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n un intento inicial de definirlo podríamos des- 
cribir el lenguaje C como un lenguaje de pro- 
gramación de propósito general muy bien adap- 
tado a la programación estructurada. 

Al decir que se trata de un lenguaje de pro- 
pósito general expresamos la idea de que no se 
trata de un lenguaje específico u orientado ha- 
cia algún tipo de aplicación en particular, como 
pueden ser los casos de los lenguajes COBOL 
(COmión Business Oriented Language) y FORTRAN (FORmula 
TRANslation) orientados a la gestión comercial y al cálculo cien- 
tífico, respectivamente. 

La idea de un lenguaje de programación de propósito 'gene- 
ral no se refiere sólo a la orientación del lenguaje hacia algún cam- 
po de aplicación, sino a la ausencia de instrucciones específicas 
para realizar determinadas tareas y, lo que es más importante, a 
la ausencia de las limitaciones (muchas veces artificiales) impues- 
tas por los lenguajes de programación tradicionales. 

Este hecho dota al lenguaje C de una potencia inusual: por 
una parte presenta un juego de instrucciones muy reducido, pero 
suficiente y fácil de aprender, y por otra ofrece muy pocas limi- 
taciones en cuanto a su utilización. 

C es un lenguaje utilizado tanto para la implementación de Sis- 
temas Operativos y Lenguajes de alto nivel, como para la realiza- 
ción de Utilidades y Programas de Aplicación. No debemos olvi- 
dar que C es el lenguaje nativo del S. O, UNIX (aproximadamente : 
un 95 por 100 está escrito en este lenguaje) ni que la mayoría 
de los lenguajes de programación soportados por UNIX (BASIC, 
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COBOL, FORTRAN y PASCAL) están íntegramente escritos en len- 
guaje C. 

Es de destacar el elevado grado de portabilidad que ofrece 
el C, permitiendo que aplicaciones desarrolladas en él sobre un 
equipo con hardware de un determinado fabricante puedan fun- 
cionar sobre equipos de otros fabricantes con la realización de 
unas mínimas e incluso inexistentes modificaciones. Esto implica 
una virtual independencia del software con respecto del hardwa- 
re y disminuye los costes de desarrollo software al ampliar el es- 
pectro de equipos sobre los que puede funcionar un determina- 
do paquete de aplicación. 

C soporta muy bien el empleo de programación estructura- 
da, admitiendo el diseño de programas mediante la realización de 
bloques cada vez más complejos y refinados (diseño "de arriba- 
abajo” o top-down) e incorporando las estructuras de control bá- 
sicas como WHILE, DO, UNTIL, SWITCH e IF-THEN-ELSE. 

Por otra parte, el lenguaje de programación C posee carac- 
terísticas de relativamente bajo nivel, como son el manejo de di- 
recciones de memoria, el tratamiento de campos de bit, el acceso 
a funciones a nivel de Entrada/Salida del sistema ogerativo (open, 
read, write, seek, close) y su implementación a base de librerías 
de subrutinas. 

La mayoría de las funciones de C están realizadas en forma 
de subrutinas escritas en el propio lenguaje C, de modo que re- 
sulta natural su enriquecimiento mediante la inclusión de nuevas 
funciones, escritas a medida que son requeridas. 

Para sentar ideas podríamos comparar en algunos aspectos 
al C con otros lenguajes de programación, como BASIC, PASCAL 
y ENSAMBLADOR, diciendo que C tiene casi la sencillez del 
BASIC, admite programación estructurada como el PASCAL y 
llega en ocasiones a un nivel tan cercano al hardware como el del 
ENSAMBLADOR. 

Como contrapartidas, C> presenta una sintaxis relativamente 
compleja y una dificultad de comprensión inicial, tal vez debida 
a la costumbre y adquisición de "vicios de programación” inhe- 
rentes al uso de otros lenguajes. 

En los primeros pasos al programar en lenguaje C el progra- 
mador se siente un poco indefenso ante la aparente falta de po- 
tencia del lenguaje, que implica el uso de funciones de librería 
para casi todo aquello que está acostumbrado a hacer mediante 
instrucciones en o:ros lenguajes. 

No deja de ser chocante para un programador enfrentarse al 
hecho de que para copiar un literal o cadena (string) sobre otro 
requiera una función de liberaría (strcpy), o que ro existan ins- 
trucciones para el manejo de ficheros indexados (C sólo puede 
tratar ficheros de acceso secuencial y relativo), o incluso que no 
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existan funciones de formateado para Entrada/Salida de datos 
(como es el caso de las máscaras de impresión, por ejemplo 
RRA HP). 

Sin embargo, resulta trivial escribir una función que copie un 
literal sobre otro, es relativamente sencilla la escritura de una fun- 
ción de formateado con máscaras de impresión, están disponibles 
comercialmente unas excelentes librerías de manejo de ficheros 
indexados, bases de datos, formateadores de pantallas y todo tipo 
de utilidades de programación escritas integramente en lenguaje 
C, y que son totalmente transportables de un sistema a otro. 

La conclusión que debemos extraer es que es mucho mayor 
la flexibilidad que se obtiene al manejar librerías de funciones 
que la que da el conjunto de instrucciones incorporadas por un 
lenguaje de alto nivel, aunque al precio de una mayor inversión 
inicial en desarrollo de software, hasta completar la escritura de 
las propias liorerías de funciones. Esto es algo que no permiten 
todos los lenguajes de programación, o al menos no con la misma 
naturalidad que el lenguaje C. 

Una de las limitaciones aparentes es su carencia de posibili- 
dades para facilitar la multiprogramación, que podríamos definir 
como la posibilidad de partir un programa único en otros varios, 
ejecutándolos simultáneamente y dotándolos de un medio que po- 
sibilite la comunicación y cooperación entre ellos. 

Sin embargo, el sistema operativo Unix suple esta carencia 
mediante las oportunas llamadas a funciones del núcleo del sis- 
tema. Allí donde no llega el lenguaje C por sí solo puede llegar 
ayudado por el núcleo de Unix, que podríamos ver como una es- 
pecie de caja negra con la única misión de activar y desactivar 
procesos (un proceso no es más que un programa en ejecución), 
realizar operaciones de Entrada/Salida, permitir la intercomunica- 
ción de procesos y tratar adecuadamente las interrupciones y con- 
diciones de excepción que se puedan presentar en el transcurso 
de la ejecución de un proceso. 

Como el sistema operativo Unix está escrito en su mayor par- 
te en lenguaje C, esto implica que las funciones primitivas del sis-' 
tema (o system-calls, pequeñas funciones aisladas cuyo conjunto 
forma el sistema operativo) puedan ser utilizadas directamente 
por los programas en C que así lo requieran sin necesidad de rea- 
lizar una interface con lenguaje ensamblador como viene suce- 
diendo en la inmensa mayoría de los sistemas operativos. 

Podemos afirmar que el sistema operativo Unix y el lenguaje 
de programación C son como dos partes de un todo indivisible: 
C es el lenguaje de programación idóneo para un entorno Unix, 
y Unix es el sistema operativo que más facilidades ofrece al em- 
pleo del lenguaje C. A través de la utilización conjunta de Unix y 
C es como se alcanza la máxima potencia y eficacia de ambos. 


En los próximos capítulos presentaremos los orígenes del len- 
guaje de programación C y haremos una descripción detallada 
del lenguaje, sus tipos de datos, estructuras de control y funcio- 
nes, con la inclusión de ejemplos en aquellos apartados donde fa- 
vorezcan su comprensión. También comentaremos algunos aspec- 
tos interesantes del compilador de C y trazaremos una panorámi- 
ca acerca del estado actual del lenguaje, sus tendencias, nocio- 
nes de C avanzado, y una breve descripción de algunas librerías 
de funciones notables. 

Todo esto con las naturales limitaciones de espacio y profun- 
didad en la exposición impuestas por el carácter divulgativo de 
este texto. Para los lectores expertos o interesados en profundizar 
en el conocimiento del lenguaje C se ha incluido una amplia bi- 
bliografía. 

Dada la escasez de libros sobre C en castellano (las pocas edi- 
ciones que existen son además traducciones) confiamos en que 
este libro les sirva de ayuda para dar sus primeros pasos en él y 
que seamos capaces de transmitir a nuestros lectores la inquie- 
tud y el deseo de profundizar en el conocimiento y dominio de 
un lenguaje tan lleno de posibilidades como es el C. 


HISTORIA DEL C 


l lenguaje de programación C fue desarrollado 
en la década de los setenta por Dennis Ritchie 
en los laboratorios Bell de la ATéT, en Murray 
Hill (New Jersey). Su historia corre paralela a la 
del desarrollo del sistema operativo Unix por 
Ken Thompson en los mismos laboratorios. 

El nacimiento de Unix tuvo lugar por la ne- 

cesidad de disponer de un sistema operativo 

con unas posibilidades similares a las de los 
grandes sistemas de la época (1970) y que pudiese funcionar en 
pequeños ordenadores de propósito general como los DEC 
PDP-11. Con esta finalidad, Ken Thompson desarrolló una primera 
versión de un minisistema-operativo al que denominó UNIX (fren- 
te a MULTICS, un s. o. de grandes ordenadores), escrito inicial- 
mente en lenguaje ensamblador. 

Posteriormente se reescribió esta primera versión en un nue- 
vo lenguaje de programación denominado “lenguaje B”, desarro- 
llado por Ken Thompson e inspirado en el lenguaje BCPL, similar 
en algunos aspectos al Pascal de Niklaus Wirth. 

Este lenguaje (B) fue posteriormente revisado por Dennis 
Ritchie, ampliándolo y obteniendo una nueva versión a la que 
denominó “lenguaje C”, desarrollado a la vez que se reescribía 
UNIX en él. 

Estaban lejos de imaginar, tanto Ken Thompson como Dennis 
Ritchie, lo que sucedería pocos años después con el sistema ope- 
rativo Unix y el lenguaje C, convertidos en estándares de la in- 
dustria de ordenadores. í 

Tras el desarrollo de las primeras versiones de Unix y C, se 
cedieron licencias de uso a distintas universidades norteamerica- 
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nas, que contribuyeron con sus mejoras y desarrollos posteriores 
a lograr una primera versión comercial del sistema operativo Unix 
hacia finales de los años setenta, 

Habrían de pasar unos años hasta que Unix y C alcanzasen 
su posición actual, favorecidos por el tremendo desarrollo del 
hardware, capaz de conseguir microordenadores de 16 bits con 
la misma o mayor potencia que los antiguos minicrdenadores de 
tan sólo una década atrás. Se hacía necesario un sistema operati- 
vo capaz de aprovechar la potencia de estas nuevas máquinas y 
la industria del software no estaba en condiciones de ofrecer nin- 
cuna alternativa válida por el momento. 

Había que conseguir un sistema operativo de fácil implermen- 
tación en un ordenador, potente en sus prestaciones, y lo suficien- 
temente estándar como para ocupar el papel que el antiguo CP/M 
había jugado en el campo de los sistemas operativos para peque- 
ños ordenadores personales. 

Tras un tiempo de pruebas empezaron a salir al mercado las 
primeras máquinas trabajando en sistema operativo Unix, causan- 
do extrañeza el hecho de que en su mayor parte (cerca de un 95 
por 100) estuviese escrito en un “extraño” lenguaje de alto nivel 
denominado "lenguaje C”. El que no estuviese escrito en su tota- 
lidad en C se debía a que era necesario codificar en ensamblador 
las subrutinas de interface con el hardware y algunos bloques de 
código por razones de eficiencia en su utilización. 

Una vez desarrollados Unix y C llegó la hora de escribir otros 
lenguajes de programación que pudiesen funcionar sobre Unix; 
fueron también desarrollados en C, lo mismo que bases de datos, 
programas de comunicaciones y todo tipo de utilidades. Sin dar- 
se cuenta, C se había convertido en una parte importantísima del 
sistema operativo Unix y prácticamente inseparable en su uso. 

Llegados al momento actual observamos que la casi totalidad 
de los fabricantes de ordenadores han adoptado el sistema ope- 
rativo Unix o derivados (como Xenix, Onyx, Zeus, Sinix, Venix, etc.) 
en sus equipos y que el lenguaje € se ha convertido en un de- 
nominador común en todos ellos. 

Por si fuera poco, el C se ha mostrado como un lenguaje de 
una elevada portabilidad en su traspaso de programas entre or- 
denadores, lo cual es una razón para su empleo por parte de los 
fabricantes de software, al ampliar el horizonte de sus productos. 

La tecnología actual ha desarrollado los primeros ordenado- 
res de 32 bits capaces de funcionar con el sistema operativo Unix, 
siendo totalmente compatibles con sus hermanos “menores” de 16 
bits, y ya se apuntan los primeros pasos hacia microordenadores 
de 64 bits, con posibilidades tan interesantes como el hecho de 
incluir procesamiento distribuido, siendo de esperar que también 
funcionen bajo el sistema operativo Unix. 
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Aparentemente estamos ante un estándar de mercado incues- 
tionable, como es el sistema operativo Unix, llamado a ocupar en- 
tre los microordenadores dotados de posibilidades multiusuario 
(varios usuarios trabajando simultáneamente en el mismo ordene- 
dor) y multitarea (un usuario puede ejecutar varios programas si- 
multáneamente) el papel de los CP/M y MS/DOS en los peque- 
ños ordenadores monopuesto. Y, por ahora, el lenguaje C es el len- 
guaje de alto nivel que mejor aprovecha las posibilidades de es- 
tos equipos. 
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ELEMENTOS DEL LENGUAJE C 


ntes de pasar a describir en detalle el C en este 
capítulo pasaremos una breve revisión a sus ele- 
mentos: tipos de datos, operadores y estructu- 
ras de control que, aunque serán objeto de es- 
tudio en capítulos posteriores, se presentan aquí 
brevemente con la finalidad de dar una peque- 
ña visión global del lenguaje y facilitar la com- 
prensión de los ejemplos posteriores. 


Tipos de datos 


El lenguaje C soporta los siguientes tipos de datos: 


— Caracteres, ya sean ASCII! o EBCDIC (según el juego de 
caracteres empleado). 

— Enteros, con o sin signo. 

— Números en coma flotante, en simple o doble precisión, 

— Funciones. 

— Arrays de cualquier tipo de variables. 

— Punteros a cualquier tipo de variables. 

— Estructuras. 

— Uniones. 
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Operadores 


Existen los siguientes operadores: 


— Aritméticos (+, — *, /, %). 

— Relacionales (>, >=, <, <=, ==, !=). 
— Lógicos (88%, [|, 1). 

— Manejo de bits (8, |, A, <<, >>, ). 
— Asignación (=, <Op>=, ++, —). 

— Condicional (? :). 

— Acceso a datos (*, €x, [], .. >). 


Estructuras de control 


Las estructuras de control disponibles en lenguaje C son: 


— Sentencias o expresiones. 

— Bloques; segmentos de programa encerrados en dos lla- 
MOS Y 

— Sentencia [f-else. 

— Sentencia switch 

— Bucles for, while, do y until. 

— faltos y etiquetas (goto, break, continue). 


Tipos de constantes 


El C admite la definición y empleo de los siguientes tipos de 
constantes: 

— Enteros: dígitos numéricos. Se interpretan como números 
decimales, a menos que su primer dígito sea un 0, en cuyo caso 
se interpretan como números expresados en octal. Por ejemplo, 
la constante 30 en decimal equivaldría a la constante 036 en base 
octal. 

— Ox u OX indican un número en base hexadecimal (em- 
plean los caracteres [0..9][a..f] o “0..9][A..F]. Por ejemplo, 30 pue- 
de escribirse como Oxl1E o Oxle, 

Las letras “1” o “L” después de una constante entera indican 
que se trata de un “long integer”, o entero largo. Por ejemplo 301 
o 30L. 

— Números en coma flotante, terminados en un "” o seguidos 
por una “e” o una “E”. Por ejemplo, 6., 5.0, 123.456e-7, 0.987E6. 

— Caracteres. Entre comillas simples ('). Por ejemplo 'a'. 

Los caracteres de control o no gráficos se representan por se- 
cuencias de escape: 1 n (retorno de carro), 1t (tabulador horizon- 
tal),1 r (salto de línea),1 0 (carácter nulo, ASCII 0), * (carácter “V”), 
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Y (comilla simple, “*”), o 1xxx, donde xxx representa un número 
votal de 3 dígitos. 
Constantes alfanuméricas: cero o más caracteres entre co- 


millas dobles (*). 
Entrada/salida simple 
La función getchar() lee el siguiente carácter presente en la 
entrada (el fichero. estándar de entrada es norma'mente el tecla- 
do). Análogamente, putchar(c) escribe el carácter "c” a la salida 
[el fichero estándar de salida es normalmente la pantalla) 
printf(formato, primero, segundo, tercero, ..) es una función de 


¡ormateado de propósito general. Los caracteres que aparecen en 
ul literal “formato” son enviados a la salida estándar, lo mismo que 
putchar, salvo que se encuentre un signo %, indicando que sigue 
una especificación de formato. Por ejemplo, %6.1f es semejante a 
P'6.1 de Portran, imprimiendo un número en coma flotante utilizan- 
do al menos 6 caracteres, con uno después del punto'decimal. 
EA PRIMO un número en coma flotante. 
Lo y %x imprimen un número entero o un carácter en de- 

C 'imal, octal O hexadecimal, respectivamente, mientras que %D, %0O 

* imprimirán un “long int” (o “entero largo”, en doble palabra), 
on decimal, octal y hexadecimal (o también la, lo y 1x). 

: imprime un único carácter. Así, printf("%c”, c) es parecido a 
putchar(C). 

: imprime una cadena de caracteres (string), terminada por 
el código ASCI 0 (carácter nulo o terminador de string). 

/, imprimirá el carácter %. 

Todas estas funciones forman parte de la denominada "libre- 
ría C estándar de Entrada/Salida”, pero no forman parte directa- 
mente del lenguaje C y serán vistas en mayor detalle al hablar de 
la librería C estándar. 
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EL COMPILADOR 


es un lenguaje compilado, a diferencia de 
otros, interpretados o seudocompilados. Va- 
mos a describir un poco lo que es todo esto. 

Los OS compilados son traducidos 
a lenguaje de máquina por un programa espe- 
cial gue se denomina compilador. Este no es 
más que un programa que se limita a analizar 
nuestro programa fuente (el original que he- 
mos escrito), comprobando su sintaxis, guar- 
( lando 1 unas tablas con las variables y direcciones que manejamos 
en él y transcribiéndolo al lenguaje interno del ordenador que es- 
lenos utilizando, obteniendo lo que se denomina "programa ob- 


El programa objeto ha de ser enlazado (linked) con una serie 
le subrutinas (como las utilizadas para gestión de memoria y pe- 
riféricos) que le permitirán convertirse en un programa ejecula- 
ole, semejante a los restantes comandos de nuestro ordenador. En 
la figura 1 podemos observar las fases necesarias para proceder 
a la compilación de un programa. 
Una vez que tenemos el módulo objeto será necesario "mon- 
larle” (linkado) las librerías necesarias para convertir el módulo 
en ejecutable. 

Los programas interpretados no siguen esta secuencia de pa- 
sos, utilizando un módulo ejecutable (denominado run-time) que 
contiene las subrutinas necesarias para proceder a la ejecución 
de! código fuente instrucción a instrucción. 

En ocasiones se procede a un análisis sintáctico previo del có- 
digo fuente; es una forma intermedia que facilita posteriormente 
la ejecución por parte del módulo de “run-time”, hablándose en- 
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Programa Fuente 


1 Compilador de i 
lenguaje ' 


j Linker SS 222 Dtros Módulos 


Programa Objeto Ejecutable 


E Figura 1.—Fases de compilación de un programa. 


tonces de programas “seudo-compilados”, a medio camino entre 
los programas interpretados y compilados; su estructura la pode- 
mos observar en la figura 2. 


¡ Módulo de A 
' run—time n | 
¡ Programa ¡ 

' Fuente Ca 


iS Figura 2.—Noción de programa «seudo-compilado». 


Ejemplos típicos de lenguajes interpretados son el BASIC y 
el PASCAL, y de lenguajes seudo-compilados, e: RM/COBOL y al- 
gunos BASIC existentes para microordenadores. 

"Un lenguaje compilado alcanza una velocidad de ejecución 
mucho mayor que la de los lenguajes interpretados o seudo-com- 
pilados, aunque ocupa más memoria al tener que incluir los mó- 
dulos que convierten al programa en ejecutable. 
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En general, un programa compilado consta de tres zonas bien 
diferenciadas: de texto, datos inicializados y stack, como podemos 
observar en la figura 3. La zona de texto se refiere al código del 
programa;los datos inicializados,a las zonas fijas de memoria des- 
linadas a contener datos que no van a ser modificados jamás, y 
ol siack, al almacenamiento de variables temporales y retornos de 
lunciones. 


Í Texto j 
i (Código) ¡ 
¡ Datos i 
' inicia- i 
1 lizados 1 
¡ Stack i 


0 Figura 3.—Aspecto interno de un programa compilado. 


Cuando la zona de texto de un programa compilado puede 
ser compartida por varios procesos que estén ejecutando simul- 
láneamente el mismo programa, entonces se habla de un "texto 
puro” o de un “programa ejecutable puro”, obteniéndose un im- 
portante ahorro de memoria central al no ser necesario almace- 
nar varias copias del programa, sino una sola compartible por los 
distintos procesos que así lo requieran. 


El compilador de C 


Para compilar programas en lenguaje C se emplea un pro- 
na denominado “cc” (c-compiler) utilizado como: 


cc nombre.c 


donde “nombre.c” representa un programa fuente escrito en len- 
guaje C; se utiliza normalmente la terminación ".c” para indicar 
este hecho. 

El compilador de C buscará en la denominada "librería de € 
estándar” aquellas funciones de las que haga uso el programa 
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“nombre.c” y las montará sobre el mismo para obtener un progra- 
ma ejecutable. 

En la llamada al compilador se le pueden “pasar” (comunicar) 
opciones cómo el indicarle un nombre de salida (opción -o) para 
el programa ejecutable (si no tomaría por defecto el nombre de 
salida “a.out”). Así 


cc -o nombre nombre.c, 


compilaría el programa "nombre.c” denominando “nombre” al pro- 
grama ejecutable, Otras opciones para el linker serían la obten- 
ción de un módulo ejecutable puro o el montaje con otras libre- 
rías distintas de la librería C estándar. Por ejemplo: 


cc -O -nv -o nombre nombre.c -lm -lcurses -ltermcap 


Sobre el sistema operativo Unix tendría el significado de com- 
pilar el programa fuente escrito en lenguaje C 'nombre.c”, llaman- 
do al programa resultante “nombre” (opción -o), empleando la po- 
sibilidad de optimización de código (opción -O), suministrando có- 
digo ejecutable puro (opción -n), con indicación de las fases se- 
guidas en la compilación (opción -v o “verbose”) y con búsqueda 
- de funciones enla librería matemática de C (opción -Im), en la li- 
brería "curses" (opción -lIcurses) y en la librería "termcap” (termi- 
nal capabilities) con la opción -ltermcap. 

La forma general de un programa en C que incluya el manejo 
de alguna función es la mostrada en la figura 4. 

El ejemplo más sencillo de un programa en C podría ser: 


main () 
£ 


print+* ("hello, world"); 


Denominamos a este programa “hola.c” y procedemos a com- 
pilarlo con la línea de comando 


cc hola.c 


Si todo ha ido bien y no nos ha faltado poner ninguna llave 
(“(” o “)”), ni el punto y coma ";”, ni los paréntesis que flanquean 
a "main" (“main()”) entonces, si ejecutamos el comando “a.out* ob- 
tendremos como respuesta el texto “hello,world” y habremos com- 
pilado y ejecutado con éxito nuestro primer programa en lengua- 
je C. 

Este ejemplo está tomado del libro de Kermigham y Ritchie 
"El lenguaje de programación C” y es típico en la mayoría de los 


Hintclude <stdio.h> -| Inclusión de ficheros de texto 
Miwline MS/DOS 3 |> y definición de simbolos mane— 
| jados por el preprocesador de 

lenguaje C. 


mb 1s =| Definición de variables 

duuble y; ¡> "globales" o comunes al 

0 =| programa principal y a 
las funciones. 


Mmiintargc, argv) =| Programa principal, con 
int arge; pa argumentos en la línea de 
vhar *arovEl; = llamada. 
/ 
int iz . Variables locales, o 
char cri0]; > pertenecientes sólo al 
float x3 =| programa principal 


CUERPO DEL PROGRAMA PRINCIPAL 
Conmdiciones, bucles, 
llamadas a funciones, etc. 


Función con argumentos 
y definición de los - 
mismos. 


Gomubla, Dy ..., nm) 
mt as 


Foat ms SS 


int iz | Variables "temporales" 
... ¡> 0 pertenecientes sólo 
| a la función 


CUERPO DE LA FUNCION 


8 0) Figura 4. —Forma general de un programa C. 


libros como el programa más sencillo y el primero que debemos 
de compilar antes de pasar a ejemplos más complicados. 

Ahora podríamos modificar el texto "hello,world” y convertir- 
lo en “hello,world|n” de modo que se produjese un salto de línea 
después de imprimir el texto. (Nota: Se trata de la barra inclinada 
«la izquierda “V”, no del signo habitual de división “/”. Si nuestro 
y clado está configurado en castellano, en lugar de como ASCII 

ISA, y pierde el símbolo, emplearemos el carácter "Ñ” en lugar 
( do A) NENE 
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Podríamos seguir esta línea de explicación describiendo 
ejemplos cada vez más complicados en los que fuésemos incor- 
porando sucesivamente los distintos elementos del lenguaje C, 
pero para eso ya están los libros incluidos en la bibliografía. Es 
proferible pasar una rápida revisión al lenguaje C y a sus elemen- 
tos, poder detenerse un poco más en aspectos como las librerías 
de funciones, la escritura de alguna función de librería, y apuntar 
algún concepto que otro de C avanzado, que es lo que veremos 
en los próximos capítulos. 


Pm. 


EL C MAS A FONDO 


Nombres de variables 


na variable no es más que una posición de me- 
moria donde almacenar un valor. Las variables 
se caracterizan por un nombre, un tipo de dato 
al que hace referencia y un tipo de almacena- 
miento, indicando de qué manera se lleva éste 
a cabo. 

Los nombres de variables consisten en se- 
cuencias de letras y dígitos. Por ejemplo: cuen- 
ll ta, importe, j5, x, son posibles nombres de va- 
rlables en lenguaje C. 

El primer carácter debe ser siempre una letra, no un dígito 
(Bj no es un nombre válido, a diferencia de j5, que sí lo es). El ca; 
Iácter especial de subrayado “_” se admite como una letra más, 
siendo muy utilizado en C como separador en nombres compues- 
los; nombres de variables válidos en otros lenguajes, como “x.max” 
o "saldo.debe” se expresarían en C como "x_max” y “saldo_debe”. 
No se puede emplear un punto “” como separador en estos nom- 
bres compuestos porque este carácter posee un significado muy 
especial en lenguaje C, utilizándose para referenciar a los elemen- 
los simples integrantes de unos tipos complejos de datos deno- 
minados estructuras, de los que ya hablaremos en su momento. Se- 
gún esto seguirían siendo válidos nombres de variables como 
"x_max”, “saldo_debe”, "xyz123_45" e incluso "_x”. 

En los nombres de variables se establece diferencia entre el 
empleo de letras mayúsculas y minúsculas. Los nombres "MA- 
YOR”, "Mayor" y “mayor” son todos diferentes. Como regla gene- 
ral, los nombres de variables propiamente dichas se referencian 
en minúsculas, reservándose las mayúsculas para los nombres ma- 
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nejados por el "preprocesador” de lenguaje C. De momento pen- 
semos en que los nombres en mayúsculas se reservan para las 
“constantes” o valores que no pueden modificarse, a diferencia de 
las variables. Ya veremos que esta distinción es sólo aproximada, 
pero nos puede servir por ahora. 

Los nombres de variables pueden tener en principio cual: 
quier longitud, aunque sólo tienen sentido los ocho primeros ca- 
racteres. Según esto, nombres como “valor_de_x” o “valor_de_y” 
harían en realidad referencia a la misma variable “valor_de”. 

Si nombres de variables van a-ser utilizados como referen- 
cias externas (es decir, si van a ser utilizadas por varios módulos! 
o funciones de un programa), puede suceder que la longitud ofec- 
tiva quede reducida a menos de ocho caracteres (normalmente 
siete). Esta reducción viene impuesta por el sistema operativo, que 
limita la longitud de los símbolos (nombres de variables) mane- 
jados por programas como el ensamblador y el montador de en- 
laces (linker). 

Como en todos los lenguajes de programación existe una lis- 
ta de palabras reservadas (keywords) que no pueden utilizarse 
como nombres de variables, pues el compilador no sería capaz 
de distinguir si realmente están haciendo referencia a una varia- 


ble o a una instrucción. Estas palabras reservadas son: 
auto else int switch 
break entry long typedef 
case enum register union 
char extern retum unsigned . 
continue float short until 
default for sizeof void 
do goto static while 
double 1f struct 


Como podemos observar se trata de una lista realmente bre- 
ve en comparación con las existentes para otros lenguajes, como 
BASIC, COBOL y Pascal, en los que pueden existir hasta unas 150 
palabras reservadas. 

En consecuencia, podemos afirmar que un programa en len- 
guaje C se limita al empleo de este repertorio de instrucciones 
tan reducido y, por tanto, fácil de aprender. La complejidad resi- 
dirá en manejar estas pocas instrucciones para obtener bloques 
de programa más elaborados (funciones) y su interconexión has- 
ta lograr programas de mayor complejidad. 
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pos de datos en C 


Una variable siempre estará asociada a un tipo de dato en 
Dt quier lenguaje de programación. El C soporta los siguientes 
lipos de datos: 


e char: consiste en un único byte, capaz de almacenar un ca- 
rácter correspondiente al juego de caracteres empleado, 
ya sea ASCII o EBCDIC. 

% int: secuencia de dígitos correspondiente a un número en- 
tero, sin parte decimal, pudiendo tener signo o no. 

e float: nómero en coma flotante y simple precisión. 

e double: número en coma flotante y doble precisión. 


El tipo char ocupa 8 bits y almacena valores en el rango de 


128 a 127, utilizando un único byte de memoria; de manera más 
¡ráfica se ve en la figura 1. 


| bo] 


128 <= char <* 127 
1 | Figura 1.—Dato tipo “char”. 


t 


Los valores del tipo int ocupan 16 bits, en direcciones conti- 
uas, y contienen valores entre -32.768 y 32,767 almacenados en 
dos bytes de memoria (Fig. 2). 


“A 768 2 int o 2= 32.767 


Ñ y Figura 2.—Dato tipo “int”. 


Los float ocupan 32 bits de memoria, estando formados inter- 
namente por un bit de signo, 8 bits de exoonente (exponentes po- 
sitivos y negativos) y una mantisa (parte fraccionaria) de 23 bits, 
proporcionando un máximo de 6 dígitos de precisión (Fig. 3). 


[se] 8 bits exp. | 23 bits mantisa tparte fraccionaria) | 


E 


| b0 | b1 | b2 | b3] 


MN F=-* 3,—Dato tipo “float”. 


Los double ocupan 64 bits de memoria, distribuidos en un bit 
de signo, 8 bits de exponente y 55 bils de mantisa (parte fraccio- 
naria), alcanzando 15 dígitos de precisión (Fig. 4). 


| sg | 8 bits exp. | 55 bits mantisa (parte fraccionaria) | 


| B0 [| b1]5b2]€3]/|b4]€5]b6]06b7 | 


0 ¡000000000000001 <= mantisa double <= 999.999.999.999.999 


t15 dígitos de precisión) 


I Figura 4.—Dato tipo “double”. 
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Al tipo int se le pueden aplicar los calificativos “short” y “long”, 
nbieniéndose los tipos de datos “short int” (entero corto) y “long 
In (entero largo, o en doble palabra). La palabra int puede omi- 
llo en estos casos, como normalmente se suele hacer. Short y 
long proporcionan enteros de diferente tamaño y precisión a los 
Ho int, Long se refiere a un doble entero (normalmente 32 bits), 
pu liendo almacenar valores comprendidos entre -2.147.483.648 y 
11,14, 483.647, mientras que short se refiere a un entero de 16 bits, 
non valores entre -32.768 y 32.767. 

E tipo int sin calificativos puede referirse tanto a long como 
a4Hhor!, dependiendo del ordenador considerado; por ello es pre- 
lorible la utilización de short o de long para evitar problemas de 
portabilidad y pérdida de precisión al pasar los programas a otro 
mdenador. 


zativo unsigned (sin signo) se puede aplicar a los ti- 
> int (1 independienter mente de short y long). El tipo 
ns ligned hace que el bit de signo pierda su significado y que los 
|imeros sólo puedan ser mayores o iguales que cero. Así el tipo 
"imsigned short” amplía la precisión de short al rango de valores 
pntre 0 y 65.535, mientras que unsigned long puede contener va- 
lores entre O y 4,294.967,295. 


Tipos de almacenamiento 


Además de un nombre y un tipo de dato, 
un tipo de almacenamiento, cons 
¡ÓN « mpilador de lenguaje € acerca del m 
iqniarle memoria a esa vanable, 


3 variables tam- 
te en una indi- 
odo como debe 


e | Almacenamiento auto (automático): indica al compilador 
que reserve una posición para la variable en la pila o stack. 
Esta posición de almacenamiento es temporal, de manera 
que cada vez que se produce una llamada a función y se 
define en ella una variable del tipo auto se emplea una nue- 
va posición de memoria para la variable, pudiendo reutili- 
zarse esa posición para contener cualquier otra variable 
después de que se haya producido el retorno de la función. 

e Static (estático): produce la asignación de una posición fija 
(estática) de memoria, de manera que cada llamada a la fun- 
ción utiliza siempre la misma posición de memoria, conser- 
vando inalterable su contenido en el transcurso de dos lla- 
madas consecutivas a la función, 

e Extern (externo): es semejante a static, con la diferencia de 
permitir que otras funciones accedan a la misma variable, 
mientras que el otro sólo permite su uso por la función en 
que fue definida la variable. 
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.. 


O Register (registro): es semejante a auto. Indica al compila: 
dor que, a ser posible, emplee un registro interno de la CPÚ 
para almacenar la variable en lugar de una posición de la 
memoria central, para disminuir el tiempo de acceso a lá; 
variables referenciadas muy frecuentemente. Los registrog 
sólo pueden contener valores enteros, estando limitado el 
número de registros utilizables a la arquitectura de la CP 
del ordenador. Muchos compiladores de lenguaje C admi 
ten el tipo register por razones de compatibilidad, aunque 
luego internamente lo ignoren. No está permitido intentar 
acceder a la dirección de memoria de una variable de tipo 
register al no tratarse de una dirección fija, a diferencia de 
las direcciones de memoria central. 


Declaración de variables 


En C, como en todos los lenguajes compilados, es necesario 
definir todas las variables de un programa (o función) antes de 
que puedan ser utilizadas. La declaración de variables consiste en 
un tipo de almacenamiento (opcional), seguido de un tipo de va- 
riable y de una lista de los nombres de variables a los que afectal 
Por ejemplo: 


auto int Xx, y, 25 
register unsigned short int 1; 
long john; 


Las variables se pueden distribuir a lo largo de tantas decla- | 
raciones como se desee. Así, las anteriores declaraciones de va- 
riables también se podrían haber escrito repartidas en varias lí- 
neas: 


auto int x3 
auto int y: 
áuto int- 2 


Si no se ha especificado explícitamente ningún tipo de alma- 
cenamiento se supone por defecto que se trata del tipo extern (va- 
riable global, o común al programa principal y a todas las funcio- 
nes), salvo que la declaración se haya realizado en el inlerior del 
cuerpo de una función, en cuyo caso se asume el tipo auto (va- 
riable local, o perteneciente sólo a la función). 

Está permitido que en una declaración se inicialice una varia- 
ble, asignándole un tipo de almacenamiento y un valor inicial en 
la misma instrucción, con sólo acompañar al nombre de la varia- 
ble de un signo igual y una expresión. Por ejemplo: 


Definirían “x” como un entero, asignándole un valor inicial de 
10, y “asterisco” como un char conteniendo un “*”, 

5 el tipo de almacenamiento es static o extern entonces la ex- 
nmaslón tiene que ser una constante, debiendo realizarse la inicia- 
linclón una sola vez antes de que el programa sea ejecutado. 

las variables de tipos automatic y register son inicializadas 

mulomálicamente por el compilador a cero (variables numéricas) 
1 al carácter nulo (variables literales o strings). 


Conversiones de tipo 

Cuando en una expresión se utilizan variables de tipos dife- 
rentes, el compilador se ve obligado a realizar una unificación de 
lipos antes de proceder al cálculo de la expresión. Las conversio- 


nos de tipo de variables implícitas en lenguaje C siguen el orden 
de precedencia mostrado en la figura 5. 


double  ¿— +$loat 


long int 


| 


short int 


char 


pe Figura 5.—Orden de precedencia de las variables en lenguaje C. 


Esto quiere decir que si, por ejemplo, en una expresión inter- 
viniesen variables de los tipos char e int, el tipo char se conver- 
liría a int antes de proceder a evaluar la expresión. Análogamen- 
lo, si los operandos fuesen int y double, se convertirían a double, 
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Esta regla se aplica no sólo a las variables y expresiones, sino tam- 
bién a los argumentos de funciones. 

Para las expresiones en que sólo aparezcan los tipos int cuan- 
do sea posible se seguirá la regla de la figura 6. 


unsigned int 


nt 


e 


 ) Figura 6.—Orden de precedencia de las variables int. 


La conversión de un número en coma flotante (float) a un en- 
tero se realiza por truncación. Para números positivos se toma el 
entero menor o igual al número en coma flotante; en cambio, para 
números negativos, C no especifica el sentido del truncamiento si 
se desprecian los decimales por defecto o por exceso. Esto tam- 
bién depende del ordenador, de modo que el fabricante elige la 
solución que le resulte más sencilla. 

Es posible forzar una conversión de tipo en un momento dado, 
mediante e empleo del operador "cast", utilizando como “(nuevo 
tipo) expresión”, así: 


int x5 
2 = (long) x; 


char s[107; 
p = (char *X) se; 


forzaría la conversión no de la variable “x”, pues "x” permanecería 
inalterado en su valor y tipo, sino de su valor al intervenir en una 
expresión, haciendo que “z” se convirtiese al valor de “x” consi- 
derado como si fuese del be long. Análogamente “p” se inter- 
pretaría como un puntero al array “s” 

El operador cast se emplea con frecuencia para realizar con- 
versiones de tipo en llamadas a funciones. Puede suceder que 
una función espere argumentos de llamada de un determinado 
tipo y que las variables que deseamos pasarle como argumentos 
no se correspondan con ese tipo; esto se soluciona con el empleo 
de cast. 
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Tipos de operadores y expresiones de asignación 


La figura 7 muestra los distintos tipos de operadores existen- 
les en lenguaje C, que pasamos a describir con más detalle a con- 
linuación. 


aritméticas ar 7] 
Aelacionaias aa] 
Lógicos Rp 

OS AO 
A A A 
Codina RES E 
a E A da 


Es Figura 7.—Operadores en Lenguaje C. 


Operadores aritméticos 


Los operadores binarios son + (suma), - (resta), * (multiplica- 
:1Ón), / (división) y % (resto módulo). También existe un opera- 
dor unario — (cambio de signo). 

El orden de prioridad de los operadores es como en mate- 
máticas: *, / y % tienen la prioridad más alta, calculados de izquier- 
da a derecha, y se ejecutan antes que el + y el - binarios. Algunos 
compiladores aplican la ley distributiva del producto respecto a 
la sama, calculando la expresión a*(b+c) como a*b+a*c, especial- 
mente cuando "a" y “c” son constantes (como es frecuente en el 
caso de subíndices de arrays). 


Operadores lógicos y relacionales 


El lenguaje C no tiene definidos tipos de datos especiales 
para las variables booleanas “cierto” y “falso”, empleándose para 
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ellas valores enteros: distinto de cero para cierto (true) y cero 
para falso (false). 

Los operadores de relación son > Anaya que), < (menor 
que), >= (mayor o igual) y <= (menor o igual). 

Los operadores de comprobación de igualdad son == (igual) 
y != (distinto de). Un error frecuente entre los programadores en 
C consiste en intercambiar el operador de asignación (=) por el 
operador de comprobación de igualdad (==), realizando una asig- 
nación cuando lo que se deseaba Eo: era una comparación 
entre dos variables o expresiones. Este no siempre provocará un 
error del compilador, resultando en ocasiones difícil de detectar. 

Los operadores conectivos lógicos son £é (and) y || (or). Siem- 
pre se evalúan de izquierda a derecha, deteniéndose la evalua- 
ción tan pronto como se produzca el fallo de una condición (para 
el and) o se cumpla (para el or), sin necesidad de evaluar la to- 
talidad de la expresión lógica, 

La prioridad de EIASAGuon más alta corresponde a los ope- 
AT relacionales (>, >=, < y <=) y a las conectivas lógicas 
(8S y |). 


Operadores de manejo de bits 


Los operadores de manejo de bits son €: (and), | (or), A (or 
exclusivo), << (desplazamiento a la izquierda) y >> (desplaza- 
miento a la derecha). Estos operadores actúan a nivel de los bits 
de una variable y sólo pueden aplicarse a variables enteras, no 
en coma flotante. También existe un operador que proporciona el 
complemento a uno de una variable, cambiando los bits a 1 por 
0, y viceversa. 

Una aplicación típica de los operadcres de manejo de bits 
consiste en su utilización para manejar flags (indicadores). Pode- 
mos tener una variable entera que almacene un conjunto de va- 
lores booleanos en forma de bits e inicializarlos y comprobarlos 
mediante el uso de máscaras conjuntamente con los operadores 
€: y |. Por ejemplo, supongamos que ERROR es la máscara 0000001, 
entonces: 


flags=flags | ERROR 
pondría a 1 el último bit, correspondiente a ERROR, de flags. 
if (flags € ERROR) 


comprueba si flags tiene puesto el último bit (ERROR) a 1. 
Hay que advertir de la posible confusión entre el iS de 
los operadores de bit “8:” y 1” y los operadores lógicos “2.8” y “1”. 
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Operadores de incremento y decremento 


El operador ++ añade 1 a su operando, que debe ser una va- 
nable, mientras que el operador -- le resta uno. 

Estos operadores de incremento (++) y decremento (--) se 
pueden utilizar en los denominados modos pre-(incremento/de- 
cremento) o post-(incremento/decremento), conforme a la situa- 
ción relativa del operador y la variable. 

Si aparece antes el operador que la variable (modo pre), en- 
lonces se modifica la variable antes de hacer uso de su valor; por 
el contrario, si el operador aparece después que la variable (modo 
post), entonces ésta se verá modificada después de ser utilizada. 
In ejemplo ayudará a comprender estos términos. 

Consideremos la variable “x”, con un valor inicial x=5, y las 
expresiones siguientes: 


Resultado 
final 


Operación Funcionamiento 


|) y=x++  Asignaa “y” el valor de “x” 
2) y=++x Suma la"x”, asignándola a “y” 


3) y=x-— Asignaa "y" el valor de "x” 


3 
4) y=-—X Resta 1 a "x” asignándolo a “y 


A 010) 0) |< 
AA 0) O) ES 


siendo; 


(1) =post-incremento =x++ 
(2) =pre-incremento =++x 
(3) = post-decremento =X-— 
(4) = pre-decremento =-—X 


Operadores de asignación y expresiones 


El operador de asignación más simple es el igual "=", que no 
be confundirse con el operador de comprobación de igualdad 
y Considerando las expresiones 


1f (a == b) 1F (xXx = y) A 


En la primera comparación (a==b) se obtendrá como resultado un 
uno (cierto) si “a” y "b” son iguales o un cero (falso) si son dife- 
rentes, sin que se vean modificados los valores de 'a” y "b” por 
el hecho de comparar sus valores. 
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En el segundo caso (X=y) estamos asignando a la variable 
*x” el valor actual de “y”, y si este valor es distinto de cero se toma 
como cierto en la expresión, con lo que estamos (inadvertidamen- 
te) modificando el valor de la variable “x”. 

El operando a la izquierda del signo igual debe ser una va- 
riable o un elemento de un array. Entonces el valor de la variable 
se reemplaza por el valor que aparezca a la derecha del signo 
igual, pudiendo ser tanto el resultado de una expresión como una 
constante o una variable. Así son válidas las siguientes asigna- 


ciones: 5 


a 


E e AYER e ddr 


También se pueden realizar asignaciones múltiples; si quere- 
mos asignar 


10; 
10; 
10; 


Nn<X Xx 
Hou 


se puede hacer como: 
4% sy =.208 105 


de modo que tanto “x”, “y” como “z” se inicializarían al valor 10. 

El valor resultante de una operación de asignación consiste 
en el nuevo valor de la variable situada a la izquierda del signo 
igual y es de su mismo tipo. Así, si “f' ha sido declarada del tipo 
float e "¡” del tipo int, la expresión “f=i” es del tipo float, mientras 
que “¡=f" es del tipo int, obteniéndose el resultado por trunca- 
miento de la parte decimal de “f”. 

Pueden emplearse los operadores, de manera que expresio- 
nes como 


i=i+2; 


también pueden escribirse como 


14=2; 
Esto es válido para los operadores +, -, *, /, <<, >>, €, y |. En 
general, si “el” es una variable, “op” un operador y "e2” una ex- 


presión, entonces 
el op=e2; . 
es equivalente a 
el = (el) oo (e2); 


con la diferencia de que “el” se calcula sólo una vez. 
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Si consideramos X = X + 1; se generará el siguiente código in- 
lerno: 


move A,x ; Carga el valor de x en el acumulador 
move B,] ; Carga 1 en el registro B 

add B ; Suma B al acumulador 

store Xx : Almacena el resultado en x 


Mientras que la expresión x++ daría lugar a: 


move A,x : Carga el valor de x en el acumulador 
incr A : Suma 1 al contenido del acumulador 
store X : Almacena el resultado en x 


Observamos que en el segundo caso se economiza una ins- 
Irucción, generándose un código más compacto y rápido. 

Los operadores “++” y "-—" equivalen a “+=1” y a "-=1”, con la 
inclusión de los paréntesis necesarios. Por otro lado 


es como escribir 


X=y+l; 
X=xX* y, 


Los paréntesis también son significativos, así 
x *=y+l; 
equivale a: 
O t 


Como ejemplo de manejo de operadores en C vamos a ver 
la función borra(s, c), que elimina todas las ocurrencias del carác- 
ler *c” en la cadena de caracteres “s”. 


/X 
* Función para eliminar el caracter(es) c 


* de la cadena de caracteres s 
*/ 


borrals, Cc) 
char s[1; 
int c; 
t 
int LL. 33 
for (im jo= 07 s[id l= *10%y i++) 
if (stil !l= c) 
s[j++] = s[ild; 
sljl = "10? 


. 
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El funcionamiento del programa es el siguiente: mediante la 
instrucción for se explora el array “s” carácter a carácter. Cada vez 
que se encuentra un carácter distinto de "c” en la posición actual 
(1), éste es copiado en la posición “j” y se incrementa “j”. Por últi- 
mo se añade el carácter nulo 10' como terminador de string. 


La instrucción s[j++]=s[i] sería equivalente a la secuencia: 


s[j]=s[i]; 
j=j+1; 


Como aún no hemos mencionado la instrucción for ni el ma- 
nejo de arrays, diremos brevemente que for se emplea para im- 
plementar bucles, constando de una expresión inicial (i =j = 0), 
. una condición de final de bucle (s[i] !'="10), que provoca la per- 
manencia en el bucle mientras sea cierta la condición y hace que 
se ejecute la acción i++, así como el cuerpo (restantes instruccio- 
nes) del bucle. 

Los arrays de caracteres s[i] y s[j] consisten en una colec- 
ción de valores de tipo char, a los que se accede por medio de 
un subíndice (“i” o “j”). Así s[0] corresponde al primer elemento, 
s[1] al segundo, y así sucesivamente. 


Expresiones condicionales 


Para calcular la mayor de dos variables “a” y “b" podemos es- 
cribir el siguiente programa: 


if (a > b) 
Z=3 
else 
2) 


que asignaría a 'z” el mayor valor de “a” o “b”. La instrucción ifin- 
cluye la comprobación de una condición lógica (a > b), ejecután- 
dose la(s) instrucción(es) siguiente(s) al if en caso de que la ex- 
presión resulte cierta, y las instrucciones siguientes al “else” en 
caso de que la condición resulte falsa. 

Otra forma de hacer lo mismo sería: 


Zz=(a>b>?ab); 


debido a que el compilador de C trata los signos “?” y “” como si 
se tratase de una construcción if-else, de manera que si se verifi- 
ca (a > b) asigna z =a (correspondiente a “?”) y z=b en caso con- 


11.3 


trario (*”), 7 
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ESTRUCTURAS DE CONTROL 


as estructuras de control son instrucciones (o. 
conjuntos de instrucciones) que sirven para di- 
rigir la ejecución de un programa, procediendo 
a la ejecución de unos u otros segmentos o mó- 
dulos del mismo de acuerdo con el resultado 
obtenido tras la evaluación de determinadas 
condiciones lógicas. Para llevar a cabo la com- 
"probación de estas condiciones lógicas existen 
h las denominadas "instrucciones de control”, es- 
elalieadas en diversos tipos de comprobaciones y acciones 
subsiguientes. 

Una sentencia consiste en una expresión seguida por un pun- 
lo y coma ", siendo normalmente una expresión de asignación, 
de incremento, decremento, o una llamada a función. 

En C el punto y coma *” es un carácter terminador de'sen- 
lencia, en lugar de un separador como es el caso de Pascal. C re- 
quiere la utilización de más "” que Pascal, pero su uso es más uni- 
forme. 

Consideremos como ejemplo el siguiente programa en C: 


while (i < 10) 13 
j=ixi; 
printec"%g dAn*, de, 3) 


3 
que podría escribirse en Pascal como: 


while i < 10 do 
begin 
mn a LI 
witeln(i:4:0, 3:4:0)5 
end 
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Las llaves “(* y. "]” se utilizan para agrupar varias instruccio- 
nes en una sola sentencia o “bloque”. Habitualmente se emplean 
para las sentencias if, else, while, do, until y fr. Las llaves “()” son 
obligatorias en la declaración de una función. 

El programa anterior imprimiría una tabla de 19 números, con 
sus valores y los de sus cuadrados. El control del programa se rea- 
liza mediante la instrucción while, que ejecuta una sentencia (el 
bloque entre llaves “([” y “)”), mientras se verifique una condición 
lógica (i < 10). 

Pascal y lenguajes similares utilizan,las palabras clave “be- 
gin” y “end”, en lugar de “(” y “)”, respectivamente. 


La sentencia IF 


Las sentencias condicionales permiten la ejecución de una 
sentencia sólo en el caso de que se verifique una determinada 
condición lógica. La sentencia "if" permite tanto la ejecución si es 
cierta una condición, como la alternativa “else” para el caso de 
que no se verifique la condición. Su sintaxis es: 


if (expresión) 
sentencia; 


O también: 


if (expresión) 
sentencial; 
else 
sentencia2; 


Es obligatoria la utilización de paréntesis para separar la sen- 
tencia “if” de la expresión. En cambio no hace falta emplear la pa- 
labra “then” como en otros lenguajes. 

Como “expresión” se puede utilizar cualquiera cuyo valor sea 
distinto de cero (cierto) o cero (falso). Así, es válida la siguiente 
condición: 


17 tstid ! 
e 


0) 
s[il; 


Mon 


que también podría haberse escrito como: 


if (s[il) 
c = stild; 


Como la sentencia “else” es opcional, un programa como 


if (nm > 0D) 


if (a > b) 

2 =aj 
else 

2 =b; 


sería ambiguo, porque la sentencia “else” podría referirse a cual- 
quiera de los dos "if. Así tanto podría significar: 


A ADA 
ie tas by A 


Zz”=A5 
> else € 
z=b;3 
3 
5 else £ 
3 /* sentencia nula *X/ 


Ls 


como: 


14 (12.00 
A Ta y Dd 
Zz=Aa; 
5 else 
3 /*X sentencia nula X/ 


.7 


Hemos colocado llaves “( )" en todos los casos, incluidos los 
de “else” con sentencia nula, para distinguir las diferencias con ma- 
yor claridad. 

El lenguaje C, como el Pascal y otros lenguajes, resuelve la 
ambigúedad asociando el “else” con la sentencia “if” más interior. 
Así el programa original es equivalente al primero, pero no al se- 
gundo. 

Una construcción de sentencias if-else empleada con bastan- 
te frecuencia consiste en: 


if (expresión 1) 
sentencia 1; 
else if (expresión 2) 
sentencia 2; 
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else if (expresión n-1) 
sentencia n-1l; 
else 
sentencia n; 


Esta construcción implementa una decisión alternativa entre 
varias elecciones mutuamente excluyentes entre sí. Básicamente 
la secuencia ifelse se refiere a una decisión entre dos opciones, 
de manera que se irá evaluando la “expresión l” secuencialmente, 
ejecutándose la "sentencia j” para la primera “)” cuya “expresión j” 
sea cierta. Todas las “expresiones J+1” y "sentencias j+1” en ade- 
lante siguientes a “j” serán ignoradas. Sólo se llegará a ejecutar la 
sentencia n” final si todas las expresiones anteriores son falsas. 


La sentencia SWITCH 


La sentencia switch verifica una elección múltiple, comparan- 
do una expresión con cierto número de constantes de los tipos 
int o char. Así, por ejemplo 


switch (c) £ 
case 
case *An”: 
case “Mt”: 
n_blancos += 1; 
break; 
default: 
n_otros += 1; 
break; 


NS Y 
: 


Ls 


La sintaxis utilizada es: 


switch (expresión) 
sentencia; 


Los paréntesis que aparecen en “(expresión)” son necesarios 
para separar “expresión” de la sentencia siguiente. Pascal utiliza. 
la partícula “of” como separador. 

Cualquier sentencia dentro de un switch puede estar prece- 
dida por una o más etiquetas “case” de la forma: : 


case constante-expresión: 


donde cada una de las etiquetas “case” de la sentencia switch pue- 
de especificar distintas constantes. 
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Al final de la cadena de “case” se sitúa (opcionalmente) una 
sola etiqueta “default”, correspondiente al camino a seguir en caso 
de que no se haya verificado ninguno de los “case” anteriores. 

La sentencia switch se ejecuta evaluando una expresión y 
vomparándola secuencialmente con cada ima de las alternativas 
"constante-expresión”, especificadas por las etiquetas “case”. 

Si el valor de la expresión es igual a una de estas “constan- 
le-expresión”, entonces la ejecución del programa se transfiere a 
la(s) instrucción(es) siguiente(s) a la sentencia etiquetada con el 
"case” en cuestión. 

Dentro de la sentencia switch, la instrucción “break” provoca 
una salida inmediata del switch. 

La sentencia switch es quizá más parecida al goto calculado 
de Fortran que al case de Pascal. Básicamente una instrucción 
switch es un goto a la etiqueta de un case determinado. A dife- 
rencia del goto calculado en Fortran, los case no tienen por qué 
comenzar en cero, o ser consecutivos. También a diferencia del 
case de Pascal una vez ejecutado un case de la sentencia switch 
el programa continuará su ejecución por el siguiente case a me- 
nos que se encuentre explícitamente una instrucción que altere 
esta secuencia (básicamente un brek, continue o return). 


Bucles WHILE 


La sentencia while tiene como sintaxis: 


while (expresión) 
sentencia; 


evaluando en primer lugar la “expresión”. Si ésta es cierta, énton- 
ces se ejecuta la “sentencia” una y otra vez, hasta que la expre- 
sión deje de ser cierta, 

Si la expresión es falsa entonces no se ejecuta la sentencia y 
se transfiere el control a la instrucción siguiente a “sentencia”. 

Es obligatorio el empleo de paréntesis “()” para separar (ex- 
presión) de la siguiente sentencia. Pascal utiliza la palabra clave 
“do” para el mismo fin. 


Bucles FOR 
La sentencia for: 


for (fexpri; expr expr) 
sentencia; 
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es muy similar a la construcción: 


E 
expri; 
while (expr2) í 
sentencia; 
expr; 


19 


3 
> 


Los “()” y *;' que acompañan a exprl, expr2 y expr3 no son 
más que separadores. La equivalencia entre ambas construccio- 
nes for y while no es exacta; la única diferencia es que while ad- 
mite el empleo de la sentencia “continue”; que la hace diferente 
del bucle for. 

La expresión "exprl” es la inicialización del bucle, “expr2” es 
la condición del bucle y "expr3” es la iteración: sentencia és la ins- 
trucción (o conjunto) a ejecutar tras cada paso. 

Habitualmente, “exprl” y “expr3" son expresiones de asigna- 
ción o funciones, mientras que “expr2” suele ser una expresión ló- 
gica o relacional. Tanto “exprl” como "expr2" y “expr3” son opcio- 
nales, pudiendo estar vacíos (o no existir), aunque los separado- 
res ";” deben estar presentes en todos'los casos. Así el bucle for 
más corto sería el bucle infinito: 


for(;;) 
senlencia; 


De este bucle sólo se podría salir mediante un salto como 
goto o return. 

Los bucles for en lenguaje C son radicalmente diferentes de 
los bucles for de otros lenguajes. Por ejemplo, los bucles for en 
Pascal se restringen al control de una variable limitada a seguir 
un conjunto ascendente o descendente de valores, como: 

for ioi= 0 to 1 do 

] 


n 
ati = 


o 


que tendría su equivalente en C de la forma: 


for (i = 03 1 < m3 i++) 
altil = 0; , 


La analogía sólo es exacta en líneas generales, porque C per- 
mite que se modifiquen las variables “i” y “1” en el interior del 
cuerpo del bucle “for”, mientras que esto no está permitido o es 


causa de error en Pascal. 


4% 


Los componentes de la sentencia “for” son expresiones arbi- 
Irarias en lugar de constantes, esto les proporciona una mayor ge- 
neralidad, permitiéndoles realizar iteraciones sobre cualquier pro- 
agresión, no sólo sobre sucesiones aritméticas. 

La ventaja de un bucle “for” sobre un “while” consiste en que 
en los bucles “for” las sentencias de control del bucle están agru- 
padas en la misma instrucción, lo que les confiere mayor claridad. 
Baste comparar el bucle “for” presentado anteriormente con 


is=Ó0 

while (1 <n) £ 
atil = 05 
ir; 


o su equivalente: 


i=0; 
while (i < n) 
ali++1 = 03 


Esta diferencia es aún más notable en el caso de que el cuer- 
po del bucle “for” sea más extenso o si en su interior aparecen va- 
rias sentencias "if". 


Bucles DO-WHILE 


Los bucles “for” y “while” siempre comprueban la condición 
de final de bucle antes de llegar a ejecutar el cuerpo del mismo. 
Sin embargo, en ocasiones se requiere que el cuerpo del bucle 
se ejecute al menos una vez y se compruebe luego la condición, 
como sucede en los bucles “do-while": 


do 
sentencia 
while (expresión); 


sentencia; 
while fexpresión) 
sentencia; 
E 


Con la diferencia de que el código correspondiente a "sen- 
tencia” no necesita duplicarse. 
Normalmente “sentencia” corresponde a una instrucción com- 
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puesta. Aunque se refiera a una instrucción simple se suelen in- 
cluir las llaves “(” y “)” de principio y final por razones de claridad. 
Un bucle como 


do 
slti++] =m2%10 + O 
while ((n / 10) > o; 


es más difícil de ver que si se escribe como: 
do £ 
s[li++] =n%10+ *0*; 
> while ((n / 10) > 0)3 


Etiquetas y la sentencia GOTO 


a” produce que la ejecución del 
:ción que sigue a "etiqueta", 

bre de variable (una cade- 
O por una letra), seguida por dos 


La sentencia “goto etiqu 
programa continúe en la i 

Una “etia 
na de letras y dígitos comer 
puntos “” : 

Una “etiqueta” puede preceder a cualquier sentencia: todas 
las etiquetas de una función deben ser distintas entre sí. 

Por supuesto, un “goto” debe referirse a una etiqueta de la mis- 
ma función. 

C no comprueba la existencia de “goto” sin sentido, como el 
salto al interior de un bucle sin pasar antes por la cabecera del 
mismo, esto es responsabilidad exclusiva del programador. 

Las “etiquetas” tienen que ser constantes, no se permiten eti- 
quetas variables. 

Aunque es posible escribir programas utilizando sólo “goto” 
e 'if” en la forma 


if (expresión) 
goto etiqueta; 


etiqueta: sentencia; 
en lugar del empleo de “else”, “switch”, etc, es más fácil, limpio e 


incluso eficiente el empleo de las estructuras de control vistas an. 
teriormente. 
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GOTOs especiales: BREAK y CONTINUE 


Los casos en que la utilización de “goto” es más eficiente y 
olara han sido resueltos mediante las instrucciones especiales 


"break” y “continue”. 
La sentencia “break” fuerza la salida inmediata de un bucle 


"lor", “do” o “while”, así como la sentencia “switch”. 


while (expresión) 
break; 


Es equivalente a: 


while (expresión) 
goto salidas 


5 


salida: sentencia; 


La sentencia “continue” produce el salto a la siguiente itera- 
ción en un bucle “do”, “for” o:"while”, Así: 


while (expresión) £ 


.. 
| continue; 


... 


Ls 
ES 


equivale a: 


| while flexpresión) 
goto sigue; 
sigue: sentenciaj 
5 


Análogamente, para el bucle for: 


for (expriz expr; expr3) < 


continue; 
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es equivalente a: 


for (expriz expr2; expr3) £ 
goto sigue; 

sigue: sentencia; 

z 


Cuándo es necesario el empleo del GOTO 


En ocasiones, es necesario salir de una estructura de bucles 
muy complicada realizada a varios niveles. Por ejemplo: 


while ((c = getchar()) != EOF) £ 
while (c == * ” plo: "An? Co== Mt?) 
if ((c = getchar()) == EOF) 


goto salida; 


, 


salida: sentencia; 


En este caso estaría justificado el empleo del goto por la di- 
ficultad de salir desde el bucle más interno hacia el exterior. 
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FUNCIONES 


as funciones se emplean para dividir los progra- 
mas grandes en otros más pequeños y maneja- 
bles. Cada función consiste en una parte del pro- 
grama, con sus propias variables, definiciones e 
incluso llamadas a otras funciones. 

Los programas escritos en lenguaje C cons- 
tan generalmente de una gran cantidad de pe- 
de queñas funciones de, a lo sumo, dos páginas de 
j Malo código fuente. Las funciones constituyen una 
manera natural de repartir un gran trabajo de programación entre 
un equipo de personas. 

Una propiedad importante de las funciones es su posibilidad 
de trabajar con argumentos de llamada y devolver valores, pu- 
diendo ser utilizadas en otros programas similares con sólo cam- 
biar los datos que figuran en la llamada a la función. b 


Definición de funciones 
La definición de una función tiene la siguiente forma: 


tipo nombre (lista-de-aroumentos) 


declaraciones-de-argumentos 
£ 
£ 


variables-locales; 
SENTENCIAS 


3 


“tipo” corresponde al tipo de dato devuelto por la función. Sl . 
no se ha especificado ninguno se supone que se trata del tipo int, 
por defecto. 
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La lista de argumentos consiste en una lista de nombres se- 
parados por comas. Esta lista puede estar vacía (función sin argu- 
mentos), pero incluso entonces siguen siendo necesarios los pa- 
réntesis “()” en la llamada a la función. 

Las declaraciones de argumentos son las declaraciones (op- 
lalivas) de las variables utilizadas como argumentos. Si no se es- 
pecifica nada se supone que se trata del tipo int, aunque es buena 
práctica definir todos y cada uno de los argumentos para dar ma- 
yor claridad a la función y a los argumentos y tipos de datos que 
requiere. » 

Las variables locales son variables que tienen validez sólo en 
el interior del cuerpo de la función. Si los mismos nombres se uti- 
lizan en otras funciones o incluso en el programa principal (basta 
con que sea fuera de la función en que han sido definidas) enton- 
ces se referirían a variables diferentes. 

El almacenamiento por defecto para estas variables es el tipo 
auto, de modo que cada vez que se llama a la función se vuelven 
a crear automáticamente todas sus variables locales. Recordemos 
que una variable es tan sólo el lugar para almacenar un valor, 
mientras que un nombre es una manera de referirse a una variable. 

Dentro de una subrutina, la sentencia 


return expresión 


devuelve inmediatamente el resultado de la expresión como el 
valor devuelto por la función. 

En C, a diferencia de Pascal y PL/1, no está permitido decla- 
rar unas funciones dentro de otras. 

Si las funciones van a devolver un tipo distinto de int, debe 
declararse el tipo en la declaración de función de la forma: 


tipo nombre-de-función (); 


Al declarar el tipo de la función, la lista de argumentos ha de 
estar vacía "()” y se debe poner un punto y coma *” al final. Si una 
función no se declara es porque se supone que devuelve un en- 
lero, como es el caso de las funciones empleadas habitualmente 
como printf y getchar, que se declaran implícitamente a través de 
su uso, 


Llamadas a funciones 


Antes comentamos que la lista de argumentos podía estar va- 
ela "()". Cuando se llama a una función los nombres de variables 
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ue aparecen en la lista de argumentos son inicializados alos va- 
lores usados er la llamada a la función; así en el programa: 


/* Eleva x a la n-ésima potencia 
X siendo x >= 0 

x/ 

power (x, n) 


£ 
% 


int i, p; 
po dz 
for (ií = di 1 <= m3 1+4+) 
P=p Xx; j 
retuntp); 
main () 
pe 
int 15 
io= power(2, 50); 
printf("i = din", 125 


» 
y 


La ejecución de la llamada a la función power como power(2, 
fi) crea dos nuevas variables con los nombres “x” y "n”, que sólo 
son accesibles por la función power. La variable "x” se inicializa a 
2, y la variable “n” a 5. Además, se crean dos nuevas variables “1” 
y "p”, también accesibles sólo por la función y que no son inicia- 
lizadas a ningún valor. : 

Las sentencias en el cuerpo de la función power se ejecuta- 
rán hasta llegar a la sentencia retum(p). Esta sentencia causará 
que el valor de la función power(5, 2) sea el valor de la variable 
"p” (en este caso, 32). La ejecución del programa prosigue enton- 
ces en la instrucción de main posterior a la llamada a la función 
power (en nuestro caso 1 = power(5, 2)). Las variables creadas para 
efectuar el traspaso de argumentos a la función, así como sus va- 
riables locales, ya no son necesarias y sus posiciones de memoria 
pueden ser reutilizadas por otras variables y funciones. 
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PREPROCESADOR MACROS. TIPOS DEFINIBLES 
Y CAMBIOS RECIENTES 


El preprocesador de lenguaje C 


1 preprocesador de lenguaje C consiste en un 
sencillo procesador de texto que realiza funcio- 
nes de inclusión de ficheros, sustitución de ma- 
cros e inclusión condicional de texto, previa- 
mente a la compilación de un programa. 

Todas las sentencias del preprocesador co- 
mienzan con el carácter “HP”, 

La sentencia del preprocesador empleada 
con más frecuencia es “tfinclude”, que realiza la 


chaióR de un fichero de texto dentro de un programa, pudien- 
do manejarse de dos formas: 


e ttinclude 'nombre-de-fichero” incluye el contenido de un fi- 
chero dentro de un programa, realizando la búsqueda del 
fichero en el directorio original donde se encuentra el pro- 
grama. Si no lo encontrase allí, entonces buscaría en una se- 
rie de directorios estándar, dependiendo del sistema ope- 
rativo. (En el caso de Unix buscaría en el directorio “/usr/in- 
clude"). 

include <nombre de fichero> es similar a la anterior, con 
la única diferencia de que el preprocesador sólo buscaría 
en los directorios estándar. Esta forma de tfinclude se em- 
plea habitualmente para incluir declaraciones y definicio- 
nes dependientes del sistema o instalación informática. 
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Sustitución de macros 
Las definiciones de la forma 
RKdefine nombre texto 


causan que todas las ocurrencias de “nombre” sean reemplazadas 
por “texto”. Nombre sigue las mismas reglas que las variables en 
C, estando formado por combinaciones de letras y dígitos. La sus- 
titución de texto no se realiza en el interior de literales entre co- 
millas (*), así si FALSO es un nombre que figura en un Hdefine, 
no sería sustituido en printf(“FALSO”). Esto es lo que se denomina 
una macro y "nombre” es, precisamente, su nombre. 

Las macros pueden utilizar argumentos si en su definición se 
acompaña un paréntesis de apertura "(” a continuación de "nom- 
bre", reemplazándose los parámetros actuales de llamada por los 
que figuran en “texto”. Si escribimos: 


define max(A, B) ((A) > (B) ? (A) 2: (B)) 
x 5 max(p+tq, r+s); 


se convertiría en 


X= ((p+0)>?(p+g): (1 + s)); 


donde observamos que cada una de las expresiones (p+q) y (1+s) 
se evalúan dos veces. Esto podría causar problemas si una expre- 
sión viene acompañada de “efectos laterales” (side effects) inde- 
seados, por ello la definición de “max” incluye paréntesis adicio- 
nales, ya que el preprocesador simplemente sustituye el texto, 
sin atender a otras consideraciones. 

Un ejemplo de efecto lateral podría ser: 


define cuadrado(x) = x X 3 
y = cuadrado(íz + 1); 


que se convertiría en: 


con un resultado final (y=2*z+1). muy distinto del deseado 
(y =(2+1)*(2+1)), como podemos observar. 


Inclusión condicional de texto 
Una línea de control como 
Hf constante-expresión 

verifica si la constante-expresión evaluada es distinta de cero. 
Hfdef identificador 


comprobaría si el identificador está “correctamente” definido en 
pl preprocesador, es decir, si previamente ha aparecido en un 
define. : 


Hifndef identificador 


chequea si el identificador no está definido actualmente en el pro- 
cesador. ; 

Las tres formas descritas van acompañadas por un texto ar- 
bitrario y la línea de control 


Fendif 


El texto puede contener adicionalmente una línea de control 
de la forma: : 


Helse 


Si la condición a comprobar es cierta, entonces se ignoran 
lodas las líneas comprendidas entre Helse (si está presente! y 
llendif. Por el contrario, si la condición es falsa, cualquier texto 
untre la condición y Helse (si existe) o Hendif (si no hay Helse) 
¡erá ignorada. 

Por convención, el preprocesador define varios identificado- 
res dependientes del sistema y de la instalación puestos a 1. Por 
ejemplo, bajo Unix funcionando en un PDP-11, el preprocesador 
define los identificadores unix y pdp11. Análogamente, un prepro- 
cesador para VAX/VMS deberá definir vax y vms. 

Esto permite al programador la escritura de programas por- 
lables incluyendo varias alternativas dependientes de la máquina 
o del sistema. Por ejemplo: 


/Kk Modo de leer una variable long de un 

* fichero binario, independientemente del 
* ordenador considerado. 

*/ 

long x = 03 


Hifdef pdpii 

x |= (lonmg)getchar() << 16; 
x (long)getchar () << 24; 
Xx 
Xx 


(1long)getchar (); 
(1long)getchar () << 8; 
Hendif 

Hif vax 

getchar (); 

getchar () << 8; 
getchar () << 16; 
getchar() <x 24; 


XXXX 


*+endi+ 

En el primer caso, el PDP-11 empaque:a un long como se: 
muestra en la figura 1 (a) y en el segundo el VAX lo hace como 
en la figura 1 (b). 


Figura 1.—Formas de empaquetar una variable long. a) PDP-II b) 
VAX. 


Consiguiendo nuestra función desempaquetar una variable 
de tipo long en cualquiera de los dos sistemas considerados me- 
diante la utilización de ttdefines combinados con las oportunas 
máscaras de bit. 


Tipos definibles 
C permite la creación de nuevos nombres de tipos de varia- 
bles. Precediendo una declaración por la palabra clave typedef, 


el nombre que figura en la declaración se considera como un nue- 
vo nombre de tipo en lugar de como una variable. Por ejemplo: 


typedef int LONGITUD; 
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hace que LONGITUD sea un sinónimo de int. La única limitación 
estriba en que 


long LONGITUD metros; 


no es una construcción válida, al no poderse añadir el calificativo 
long al typedef, a diferencia de 


long int metros; 
que sí es válida. Un ejemplo más útil podría ser: 


typedef char *Xstring; 
maintargc, arav) 


int argc; 
strina *Xargv; 


y un ejemplo algo más complicado: 


typedef struct _tnode £ 
string word; 
int count; 
struct _tnode *left; 
struct _tnode *right; 
3 tnodez 


Aunque tiene sus limitaciones, pues 


r 


typedef struct ( 
strina word; t 
int count; 
tnode Xleft; 
tnode kright: 

Y tnodez 


no funcionaría, al aparecer tnode en la definición del typedef y for- 
mando parte de sus componentes. 

Typedef no define ningún tipo nuevo de datos, sólo cambia 
de nombre a los tipos de datos. Una variable declarada como del 
lipo “string” sería lo mismo que si se hubiera definido del tipo 
"char *”, y el compilador de lenguaje C no establecería diferencia 
entre ellas. 

Los tipos definibles pueden ser muy útiles para escribir fácil- 
mente declaraciones de tipos complicados, difíciles de compren- 
der, como puede ser el caso de: 


typedef int (* PFDO; 


Crea el tipo PFI como un “puntero a una función que devuel- 
ve un entero”. Comparemos las expresiones 


PFI stremp, numcmp, swap; 
con 
int (* stremp)O, (* numcmpJO, (* swap); 


Podemos apreciar la ventaja de introducir un tipo definible, 
obteniendo una expresión más breve y fácil de comprender. 

Los tipos definibles suelen emplearse para facilitar la porta 
bilidad. Por ejemplo, si una variable necesita ser declarada como. 
un int en una determinada máquina y como long en otra, las lí- 
neas de control del preprocesador y los typedefs pueden em- 
plearse para definir un nombre que pueda funcionar adecuada- 
mente en cada una de las máquinas. 

Aunque, en la práctica, si una variable ha de ser definida como 
long en alguna máquina sería razón suficiente para declararla 
como long en todas las demás. 


Cambios recientes en C 


Algunos compiladores recientes de C reconocen la palabra 
“void” como un tipo sin valor alguno. Evidentemente, no se pue- 
den declarar variables del tipo void, pero sí declarar que una fun- 
ción devuelve un tipo void para aquellos casos de funciones que 
no devuelven ningún valor (si el compilador se encontrase pos- 
teriormente un return, generaría un error). : 

Análogamente, se puede emplear un cast con los tipos void, | 
para indicar que deliberadamente estamos definiendo una función 
que no devuelve valores. Por ejemplo: 


(void) push(pop() + pop()): 


definiría la función push como una función que no devuelve va- 
lores, 

C admite también los tipos numerados, semejantes a los que 
existen en Pascal. Un ejemplo podría ser: 


enum color (rojo, verde, azul ); 
La sintaxis y el empleo de los tipos numerados son similares 


a los de las estructuras, con la diferencia de que no pueden esta- 
blecerse miembros. 
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El primer elemento de un tipo numerado es equivalente a la 
vonstante O, el siguiente a 1, y así sucesivamente. 

A un elemento de un tipo numerado se le puede asignar un 
determinado valor acompañando al nombre del mismo por un sig- 
no igual "=" y una expresión constante. Los elementos siguientes 
continuarán la sucesión de valores a partir del primer elemento 
signado. Por ejemplo: 


enum keywords £ 
IF = 0200, 
ELSES 
WHILE, 
UNTIL, 
DO, 


3; 


Los tipos de datos numerados y sus elementos deben poder 
distinguirse entre sí, así como de las restantes variables “norma- 


les” y nombres de funciones (igual que la diferencia que debe ha- 
ber entre las estructuras y los nombres de sus miembros). 
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PUNTEROS Y ARRAYS 


n puntero consiste en una pseudo-variable que 
dará la dirección de la variable asociada. Los 
punteros (como los goto) son una instrucción 
de muy bajo nivel, y puede ser fácil llegar a 
abusar de su empleo, pudiendo casi siempre 
ser sustituidos por otras construcciones alter- 
nativas, como arrays, y llamadas por referencia. 
El operador $ proporciona un puntero a 
3 - una variable. Por ejemplo, si “x” es una varia- 
ble, entonces é:x es un puntero a “Xx”. 
El operador unario * accede a la variable apuntada por un 
puntero, así: 


es un equivalente un poco oscuro de: 


Ha Yi 
10; 
y +45 


Pp) 
Uhr 


Para declarar una variable (px) que contenga un puntero a 
un entero se hace: 


int *px; 
El aspecto de la declaración de px explica de alguna manera 


su utilización. La expresión *px apunta a un valor del tipo int. 
Del mismo modo a como hemos definido px como un punte- 
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ro a un entero, podrían definirse punteros a otros tipos de varia: 
bles, así: 


double *dp; 


Las expresiones *px y *dp tienen valores de los tipos int y 
double, respectivamente, y pueden ser utilizadas en otras expre 
siones. 

El siguiente programa, una vez compilado y ejecutado, nos 
mostrará con valores numéricos el resultado de la utilización d , 
los operadores * y €. 


main () 


£ 


int xs, Xpx; 
double *Xdp, d; 
px = lo; 


printf(í%kpx = 4d x = din", Apx, x)3 
print+("*xdp = Y%d d = ZfAn", *dp, dd); 


a 
+ 


Los operadores € y * tigenen mayor prioridad de ejecución 
que los operadores aritméticos, así: 


YA PERL 


toma el valor de la variable apuntada por px, le suma uno y asig- 


na este valor a “y”, 


Como un puntero consiste en la dirección de alguna variable, 
una expresión precedida por un * también será una variable, de Í 
modo que también podrá intervenir en expresiones de asigna- 
ción. Por ejemplo: 


int Xa Xpxs 


x= 05 
px o = lx; 
Xpx = 1; 


printf("x = %d kpx = %dWn", x, px); 


Las expresiones con un * pueden también utilizarse con ope- 
radores de incremento y decremento. Así obtendríamos expresio- 
nes como: 


Xpx += 15 
Ho 
(kpx)++z 


que suma “1” al valor apuntado por “px” 
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Punteros y argumentos de funciones 


C pasa los argumentos a funciones utilizando "llamadas por 
valor”, de forma que no hay manera directa, tras una llamada a 
una función, de que ésta pueda modificar las variables que le han 
ui¡do pasadas como parámetros. En lenguajes como Fortran o Pas- 
cal, que utilizan “lamadas por referencia”, una función sí puede 
modificar los valores de sus argumentos de llamada. 

Consideremos la función swap, codificada en Pascal, que in- 
lercambia los valores de dos variables: 


procedure swapívar Xy Y 3 integer); 
var 
temp : inteuer; 
begin 
temp 1= x5 
4 O= yr 
y ou= temp; 
end 


Y la misma función codificada en Fortran: 


SUBROUTINE SUAP(X, Y) 
INTEGER X, Y, TEMF 
TEMP = X 

NY 

Y = TEMP 

RETURN 

END 


Si ahora transcribimos literalmente esta función a lenguaje C, 
obtendremos una función de aspecto parecido, pero totalmente 
inefectiva: 


swap (Xx. y? 
int Xi Yi 


int temp; 


En lugar de esta versión podríamos escribir una función en 
C que intercambiase los valores de dos variables pasando como 
argumentos punteros a las variables en lugar de las propias va- 
riables: 


swapipx, py) 
int kpx, Apyr 
As 
int temp; 
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Xpx = *Xpy; 


Lo 


y probarla mediante el siguiente programa principal: 


maint) 


int a, b5 

a = 13 

b= 2; . 
printfí"a = Z/d b = ZdiWn", a, b?; 
swap (La, 2Lb); 

printfi"a = Zd b = AdAWn", ay b); 
return(0):; 


> 


De hecho, los lenguajes de programación que pasan “por re- 
ferencia” sus argumentos a una función en vez de “por valor”, esen- 
cialmente lo que hacen es insertar un € antes de las variables en 
la lista de argumentos, y tratar los argumentos de la función como 
si fuesen punteros a variables, precediendo siempre con un * las 
Operaciones en el interior del cuerpo de la función. 

La llamada por referencia se emplea a menudo en funciones 
que realizan "retomos múltiples”, es decir, funciones que necesi- 
tan devolver una indicación de error además de un valor, modi- 
ficando variables globales del programa que llamó a la función. 


Arrays 


Un array es un conjunto de variables del mismo tipo a las que 
se puede acceder directamente. Por ejemplo: 


int a[10); 

declara “a” como un array de 10 variables de tipo int. El índice del 

array empieza siempre en 0. Así el primer elemento de “a” es a[0], 

el segundo a[ 1], y así sucesivamente hasta a[9]. : 
La expresión utilizada para el índice de un array es una ex- 

presión arbitraria, tanto constante como variable, del tipo int. 

"  Elacceso a todos los elementos de un array se realiza habi- 
tualmente por medio de un bucle for. Por ejemplo, para imprimir 
cada elemento del array definido anteriormente se podría utilizar 
la siguiente sentencia: 


for (i= 03 i < 103 i++) 
print+("ZdWn", ali); 


C implementa internamente los arrays asignando a sus ele- 
mentos posiciones consecutivas de memoria, tratando el array 
como si fuese un puntero a la primera variable del mismo. En el 
contexto del ejemplo anterior, C definiría el valor de la expresión 
w+i lo mismo que si fuese ¿a[i]. , 

Esta operación de sumar un entero a un puntero se puede ge- 
neralizar a punteros arbitrarios y no sólo a los punteros que apun- 
tan al principio de un array. De hecho, C define una expresión de 
la forma a[i] como idéntica a "*(a+)”. A ; 

C define la sustracción entre dos punteros (llamémosles pl y 
p2), del mismo modo que p2 es igual a 


epl[pl - p2] 


Por ejemplo, pl - p2 es el número de elementos que hay en- 
lre pl y p2. Si se restan dos punteros que no apunten al mismo 
array, el resultado obtenido será en general impredecible. 

También pueden realizarse comparaciones entre punteros. 
Sin embargo, a menos que los punteros apunten al mismo array, 
sólo pueden realizarse comparaciones de igualdad (== y !=). 

Como € no pone ningún cuidado en comprobar el rango de 
los índices que maneja es relativamente fácil "pasarse de una va- 
liable", manejando arrays y punteros, e intentar alcanzar una di- 
rección no permitida, pudiendo llegar a invadir la zona de alma- 
cenamiento destinada a otras variables. Sin embargo, esta ausen- 
cia de chequeos tiene sus ventajas en cuanto a la sencillez de rea- 
lización de un compilador de C y la posibilidad de trabajar con 
direcciones de máquina a bajo nivel. 

A continuación vamos a presentar varias funciones de la li- 
brería de C estándar escritas de dos maneras diferentes: emplean- 
do arrays y empleando punteros. . 


/* Función strcpy: copia el literal s2 
X sobre el literal si 
*/ 
strcpy(sl, s2) 
char s1[J, s2(1; 
z 
int i; 
for (ií = 03 si[il = s2[113 i++) 


Funcionamiento: va copiande s2 sobre sl carácter a carácter 
mientras s2[i] sea distinto de cero (nulo). 


/* Función strlen: devuelve la longitud 
kx del literal s (calculada como el número 
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* de caracteres distintos de nulo) 
K/ 
strlents) 
char sC1; 
at As 
for ín = 045 sitnl3 n++) 


returntn); 


a 


Funcionamiento: mientras s[n] sea distinto de cero (nulo) va 
incrementando la variable “n”, igual a la longitud del string. 


/* Función strcmp: compara dos literales 
* si y s2, devolviendo >0 si si<s2, O si 
* si=s2, y <0 si si<s2 
x/ 

strcmpísi, s2) 

char s1[1, s203; 


£ 


int i = 05 
while (s1[i] = s2[i]) 
14 (sILi++J == x0”) 
returntO); 
returnís1fiJ —- s2[i1); 


Funcionamiento: va comparando sl] y s2 carácter a carácter. 
Mientras van coincidiendo los caracteres comprueba si se ha al- 
canzado el final de s1, en cuyo caso las cadenas sl y s2 serían igua- 
les, devolviendo un 0 la función. Si no es nulo se devolverá un va- 
lor >0 si es s1[i]>s2[i] o <0 si s1[1]<s2[i]. 

Estas mismas funciones se podrían haber escrito empleando 
punteros en vez de arrays, como: 


strcpy(ísl, s2) 
char Yi, =2; 
t 


while (Xsl++ = ks2++); 
> 


Funcionamiento: mientras s2 apunte a un carácter distinto de 
nulo, copia el valor apuntado por s2 sobre el valor apuntado por 
sl, e incrementa ambos punteros (++). 


strien(s) , 
char *Xs; 
X 
char Xp = s3 
while (*p) 
p++; 
return(p-s); 


4, 
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Funcionamiento: inicializa un puntero al principio del string. 
Mientras este puntero no apunte a un nulo lo va incrementando y 
al final retorna p-s que es el número de caracteres avanzados por 
el puntero y, por tanto, la longitud del string. 


strempísi, s2) 
char *si, ks2; 


for (3 *si == *Xs2; s2++) 
i4 (ksi++ == M0) 
return(t0)3 


return(ksi - Xs2)5 


Funcionamiento: mientras los valores apuntados por sl y s2 
sean iguales, va desplazando ambos punteros, comprobando en 
cada momento si se alcanza el final de sl, en cuyo caso ambas ca- 
denas serían iguales y la función devolvería un 0. Si sl no apunta 
a un nulo, se devuelve la diferencia entre los valores apuntados 
por sl y s2. 


Arrays multidimensionales 


C permite la definición de arrays de cualquier tipo, no sólo 
de los tipos básicos. 

En particular, los arrays multidimensionales son justamente 
arrays de arrays. Por ejemplo: 


double a[5][5]; 


1 
define “a” como un array de 5 x 5 elementos en coma flotante y 
doble precisión. Esta declaración es semejante a 


var 
a : array[0.4.0.4] of real; 


en Pascal, o a 
DOUBLE A(5,5) 


en Fortran. i 

En C un array de dos dimensiones se ve como Un array de 
arrays, al igual que en Pascal. Es más, ali] es. un array de 5 ele- 
mentos double. Sin embargo, a diferencia del Pascal, no puede 
abreviarse el doble subíndice. 
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E 


double alS1[(5]; 

LOL Le 3 

for (1 = 0713 053 1) 
for (j = 03 3 < 53 j++) 
; aliJ(jJ = 03 


Este programa serviría para inicializar a cero todos los ele- 
mentos del array “a”, mostrando el manejo de subíndices de arrays. 


Arrays como argumentos de funciones .. 


€ convierte siempre una referencia a un array en un puntero 
al primer elemento del array. El compilador de C trata los argu- 
mentos declarados como arrays lo mismo que si hubiesen sido de- 
clarados como punteros al elemento base del array. Así: 


strlentís) 
char sl; 
k 


2 
y 


es idéntico a 


strlents) 


char *Xs; 


ti 


5 


Declarar los argumentos como arrays o punteros es más bien 
una cuestión de estilo, aunque es recomendable declarar el argu- 
mento como un array si la función maneja índices de arrays, o 
como un puntero si accede al parámetro por medio de un *. 

El límite de dimensión más a la izquierda de un array puede 
estar vacío en la declaración de un array como argumento. Su- 
pongamos que queremos pasar el array “pantalla” como un argu- 
mento a la función “f”, que debería definirse como: 


f (pantalla) 
char pantallalLINEASICCOLUMNASI; 
E 


% 


> 


Ff(pantalla) 
char pantalla 1[ COLUMNAS 3; 
£ 


A 
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o incluso como 


f (pantalla) 
char (Xpartalla)[2OLUMNASI; 
E 


> 
> 


La última declaración establece explícitamente que “pantalla” 
es un puntero a un array de tantos caracteres como COLUMNAS 

Hay que destacar que la función “f" debe tomar el número de 
columnas de la matriz pantalla, pero no necesita conocer el nú- 
mero de filas. C es más flexible que Pascal (en él hay que decla- 
rar todos los límites de un array pasado como argumentoo de una 
función), pero menos que Fortran (solamente necesita declararse 
la dimensión, por ejemplo, “pantalla” es un array de dos dimen- 
siones). 


Comparación entre arrays de punteros y arrays multidimensionales 


Existe una diferencia importante entre un array de punteros 
y un array bidimensional. Por ejemplo, dadas las declaraciones 


char klineptriLINEASI; 
char lines[LINEASI[MAXIMO]; 


El empleo de lineptr y lines parece similar en que lineptr[1][0] 
y lines[1][0] referencian ambos a un único carácter (presumible- 
mente el primer carácter de la segunda de un grupo de líneas de 
texto). Sin embargo, lines es un auténtico array, conteniendo Ll- 
NEAS * MAXIMO caracteres. Como lines[i] es un array de MAXI- 
MO caracteres, se accede a lines[i1[] multiplicando "j" por MAXI- 
MO y sumándole 'i” para determinar a cuál de los caracteres LI- 
NEAS * MAXIMO debería acceder. 
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ESTRUCTURAS 


na estructura es un conjunto de variables de 
los mismos o diferentes tipos. Las estructuras 
son semejantes a los registros (records) de 
Pascal. 

Los elementos integrantes de una estruc- 
tura se denominan miembros. Pascal llama 
campos (fields) a los elementos de una estruc- 
tura, pero C reserva este nombre para referir- 
7 se a los miembros particulares de estructuras 
que permiten manejo de bits. 

La declaración de una estructura se hace como: 


struct y í 


miembros-y-declaraciones t 
A 


utilizándose como un tipo de dato más en declaraciones de tipo 
de variables. 

“x” es el nombre de la estructura. Las referencias posteriores 
a la misma estructura pueden omitir las llaves (()) y la declara- 
ción de sus miembros. 

Los miembros y declaraciones incluidos en la estructura son 
como cualquier declaración de variable de las vistas hasta ahora, 
con la única diferencia de que no puede especificarse un tipo de 
almacenamiento (auto, static, register, extern). Por ejemplo: 


struct fecha £ 
int dia; 
int mes; 
int año; 
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struct persona t 
char nombrelTAMAÑO1 3; 
char direcciontTAMARÑOZI; 
long cod_postal; 
long ss_num; 
double sueldo; 
struct date nacim_fecha; 
struct date incorp_t+techa; 
+; 


sería el equivalente a los records de Pascal: 


type 
date = record 
dia: 1..31;3 
mes: 1.,12; 
año: integer; 
end; 
person = record 
nombre: arrayLi..TAMARÑOD17J of char; 
direccion: arrayll..TAMAÑOZ3 of char; 
cocd_postal: integer; 
53_num integer; 
sueldo: real3 . 
nacim_fecha: date; 
incorp_fecha: date; 
end; 


En este ejemplo definimos una estructura “fecha” contenien- 
do tres variables enteras: día, mes y año, y definimos una estruc- 
tura “persona”, conteniendo dos arrays de caracteres (nombre y 
dirección) cod_postal de tipo long, ss_num de tipo long, sueldo 
de tipo'double, y dos miembros consistentes en estructuras "fe- 
cha”, correspondientes a nacim_fecha e incorp-fecha. 

Para acceder a un miembro de una estructura se utiliza el ope- 
rador *”, como ya anticipábamos al hablar de las variables en C 
y sus tipos. Se recurre a él en la forma: 


estructura-variable.miembro 


Por ejemplo, para declarar una variable "d” como una estruc- 
tura “fecha” e inicializarla a la fecha 1 de junio de 1986, se haría 
del modo siguiente: 


struct fecha d; 
d.dia = 
d.mes = 6; 
d.año = 


Como podemos observar er. la declaración de la estructura 
persona las estructuras se pueden encadenar, pudiendo utilizarse 
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como miembros de otras estructuras. Si se define "emp” como una 
estructura persona: 


struct persona emp; 
entonces 
emp.acim_fecha. año 


se referirá al año de nacimiento del empleado en cuestión. 

Una estructura externa o estática se puede inicializar acom- 
pañando el nombre de la estructura por una lista de inicializado- 
res rodeados por llaves (()). 


struct fecha d = (1, 6, 1986); 


Declara “d” como una estructura fecha, e inicializa d.día a 1, 
d.mes a 6 y daño a 1986. 


Operaciones sobre estructuras 


En implementaciones antiguas de C las únicas operaciones 
permitidas sobre las estructuras eran el acceso a sus miembros 
(mediante un “”), o el acceso a su dirección (mediante un g). 

Los compiladores más recientes permiten la asignación de va- 
riables estructura, incluyendo el paso de estructuras como pará- 
metros y devolviendo estructuras como valores de funciones. 

No está permitido realizar comparaciones de estructuras. 

Muchos compiladores que admiten la devolución de estruc- 
turas lo hacen de manera no-reentrante, dejando el valor de la es- 
tructura devuelta en una variable estática en lugar de en el stack, 

Las estructuras automáticas no pueden inicializarse a pesar 
de que inicializar variables automáticas es equivalente a realizar 
una asignación. En el S.O, Unix se obtendría el mensaje de error 
“No auto aggregate inicialitation” como respuesta a este intento. 


Punteros a estructuras 


Debido a las restricciones impuestas a las variables de tipo 
estructura normalmente se suelen utilizar punteros a estructuras 


para realizar el traspaso de estructuras como argumentos de fun- 
ciones: 
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Incluso si se permiten asignaciones de estructuras, pasar una 
estructura como argumento puede no ser muy buena idea en oca- 
siones; por ejemplo: 


define LINEAS 24 
$Hdefine COLUMNAS BO 


struct ventana £ /X*X una parte de una pantalla X*X/ 
char lineas [LINEASICCOLUMMASI; 


int ult_col; /k esquina superior X/ 
int ult_filaj /%* esquina superior */ 
int nlineas; /* número de lingas X/ 
int ncols; /* número de columnas *X/ 


3 pantalla; 


inicializa() 

t 
pantalla.ult_col = 0; 
pantalla.ult_fila = 0; 
pantalla.nlineas = LINEAS; 
pantalla.ncols = COLUMNAS; 


borra_pantalla(pantalla.lineas); 
muestra(pantalla); 


La llamada a "muestra" (definida en otro lugar) copia la tota- 
lidad de la estructura en el stack. Esta estructura contiene LINEAS 
* COLUMNAS = 24 x 80 = 1920 caracteres, así como algunas va- 
riables enteras. Compárese esto frente a la alternativa de pasar 
como argumento un puntero a la estructura, que requeriría tan sólo 
una variable entera (la dirección de la estructura). 

Hay que destacar que pantalla.lineas es un array, y aquí se pa- 
saría un puntero al primer elemento de este array en la llamada 
a la función borra_pantalla. 

Los punteros a estructuras se declaran de la manera habitual, 
como los restantes punteros: 


/*X extrae elementos de la estructura 
X persona y los imprime 
x/ 
ppersona(p) 
struct persona *Xp; 
xl 
print*("“Nombre:z Zsin", p-+nombre);z 
printf("Dirección: Zsin", p->*direccion); 
print*+ ("Código Postal: Zdin", p->cod_postal) 
print* ("Código de la S.S.: Zdin", P=->ss_num)' 
printfAX"Sueldo: %.2f1n", p-*sueldo); 
print+f("Fecha de nacimiento: %d %d Z%din", 
p->+nacim_fecha.dia, 
p->nacim_fecha.mes, 
p-3nacim_fecha.año); 
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printf ("Fecha de incorporación: %d %d Z%din", 
p->+incorp_fecha.dia, 
p->incorp_ fecha. mes, 
p->incorp_fecha. año); 


La construcción 

puntero-a-estructura -> miembro 
es equivalente a: 

(kpuntero-a-estructura) -. miembro 


El operador "-->" consiste en un signo menos "—" seguido por 


el signo mayor 
Los paréntesis en “(*p).nombre” son necesarios porque el or- 
den de evaluación del operador "." (miembro de una estructura) 
es mayor que el del operador "*" (valor apuntado por). Así 
*emp.nombre es equivalente a emp.nombre[0]. 
Los operadores ".' y "->" tienen mayor prioridad que los ope- 


radores aritméticos. Así pues, dada la declaración: 


struct ( 
int x3 
int y; 
3 Xp: 


la expresión 


tp 
t 


incrementaría el valor del miembro “x” de la estructura apuntada 


por “p”, en lugar de incrementar el puntero "p”. 


Análogamente: 

Xp->y; obtiene el valor de 'y apuntado por “p” 
Xp->y ++ idem., incrementando p->y 

(kp->y) ++ incrementa la 'y' apuntada por 'p' 
Xp++->y5 incrementa 'p' tras buscar p->y 


Del mismo modo que C permite arrays de arrays como un 
tipo más de variables, también permite la definición de arrays de 
estructuras. 
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Campos 


Dentro de la declaración de una estructura puede aparecer 
un miembro seguido por un carácter de dos puntos "” y una ex- 
presión constante, denominándose entonces campo (field). 

El tipo del miembro debe ser alguno de los tipos enteros 
(char, int, short, long o unsigned); la expresión constante especi- 
fica cuántos bits van a utilizarse para ese miembro. 

Los campos de bit son una alternativa a las operaciones ex- 
plícitas de desplazamiento y máscaras de bits. Consideremos el 
programa: 


define KEYWORD 01 
tidefine EXTERNAL 02 
define STATIC 04 


int flags; 


flags |= EXTERNAL | STATIC; 


if (flags $ KEYWORD) € 


Esto mismo se podría también haber escrito como: 


struct 4 
int is_keyword:1; 
int is_extern:1; 
int is_static:1l; 
> flags; 


flags.is_extern = flaos.is_static = 1; 


if (flags.is_keyword) 


Desafortunadamente, el empleo de campos de bit en C tiene 
algunos inconvenientes: 


e en las versiones antiguas de compiladores de C, todos los 
campos tienen que ser cantidades sin signo 

e no pueden emplearse datos binarios empaquetados de una 
máquina a otra, pues el empaquetado es dependiente de 
la máquina al no especificarse si la asignación de bits se 
hace de izcuierda a derecha, o de derecha a izquierda; * 

e los campos de bit no son exactamente variables, en el sen- 
tido de que no pueden definirse punteros a campos de bit, 
ni arrays de campos, aunque, por supuesto, sigue siendo vá- 
lido definir punteros a estructuras que contengan campos 
de bit. 
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Uniones 


Una unión consiste en una variable que puede contener va- 
lores de diferentes tipos, en lugar de un solo tipo como las varia- 
bles “normales”. Las uniones son una de las alternativas para sol- 
ventar la falta de comprobaciones de tipo del lenguaje C en tiem- 
po de ejecución. 

La sintaxis de la declaración de una unión es semejante a la 


declaración de una estructura sin más que cambiar la palabra es- 
tructura por unión. 


struct /* entrada en la tabla de símbolos */ 
char *Xname; 
int type; 
union € 
int u_ival; 
float u_fval; 
char *X_sval; 
7 uval; 
3 symtabiNSYMI; 


Las variables symtab[i].uval tienen capacidad suficiente para 
contener variables de cualquiera de los tipos int, float o char, sien- 
do responsabilidad del programador tener el cuidado necesario 
para controlar los tipos adecuados de las variables contenidas en 


una unión. 
La unión definida anteriormente permitiría el acceso a sus dis- 


tintos elementos como: 


switch. (symtablil.type> € 
case INT: 4 
printf ("%dAn", symtablilJ.uval.u_ival); 
break; 
case STRING: 
orint+("Z%sin", symtabliJ.uval_kval); 
oreak; 
case FLOAT: 
orintf("%fAn", symtablild.uval_fval); 
break; 


siendo similar al programa en Pascal: 


type symtab = 
record 
name : array[l1..10] of char; 
case stype : integer of 
int : ( ival : integer ); 
float : (fval : real ); 
string : (sval : array[l1..20] of char ); 


end; 
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var 
symtab : arrayli..nsymbolsl of symtab; 
case symtabliJ.stype of 
int : witeln(symtablil.ival); 
float : writeln(symtab[lil.fval); 
string : writeln(symtablil.sval); 
end 


La enorme similitud entre ambos programas puede aún au- 
mentarse mediante el empleo del preprocesador de C, emplean- 
do Hdefines que hagan más manejables los- nombres de variables: 


define ival uval.u_ival 
define fval uval.u_fval 
*+define sval uval.u_sval 


Las operaciones que pueden realizarse sobre las uniones son 


las mismas que las permitidas sobre las estructuras acceder a uno 
de sus miembros y acceder a la dirección de la unión. 
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ENTRADA/SALIDA EN LENGUAJE C 


l lenguaje C no incorpora directamente instruc- 
ciones para la realización de las operaciones de 
Entrada/Salida, implementándola a través de li- 
brerías y funciones. 

Existe un conjunto de funciones de E/S que 
es portable a un gran número de sistemas ope- 
rativos (UNIX, VMS, MS-DOS, CP/M, ..), imple- 
mentadas en la denominada librería de E/S es- 
“tándar. Cualquier programador de C debe co- 
nocer los nombres y argumentos de estas funciones, consideran- 
do que van a ser las mismas en cualquiera de los sistemas ope- 
rativos mencionados. 

Además se incluyen funciones para permitir el interface di- 
recto con las propias funciones de E/S del sistema operativo. 


Los ficheros en la librería estándar de E/S 


ones necesarias para el manejo de la librería es- 
1 accesibles mediante el empleo de un Hinclude; 


Las declaraci 
tándar de E/5 sor 


Hinclude <stdio.h> 


Este fichero contiene las declaraciones de variables y defini- 
ciones de macros necesarias. : 

En algunos sistemas puede llegar a ser necesario notificar ex- 
presamente al compilador la inclusión de la librería estándar. Por 
ejemplo, en los antiguos sistemas Unix Versión 6 se requería la in- 
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clusión de la opción “-1S” al compilar. Esto ya no es necesario en 
los sistemas Unix Versión 7 actuales o en sus derivados. 
Internamente, la librería estándar de E/S almacena la informa- 
ción correspondiente a cada fichero abierto en una estructura, 
identificando a los ficheros por medio de punteros a estructuras. 
El fichero <stdio.h> define FILE como una de estas. estructuras. 


Apertura de ficheros 


Antes de que se pueda acceder a un uchero éste debe ser 
abierto mediante la función fopen: 


e 


FILE *xfopen(name, mode) /* Abre un fichero */ 
char *Xname; /* Nombre del fichero a abrir */ 
char *Xmode; /X Modo de acceso al Fichero */ 


El primer argumento de fopen es el nombre del fichero, en- 
viado como una cadena de carac:eres. El formato del nombre de 
fichero es dependiente del sistema operativo. Sin embargo, mu- 
chas (no todas) las implementaciones de la librería C estándar de 
E/S trasladan los nombres de ficheros especificados como en Unix 
al formato requerido por el sistema operativo. 

Por ejemplo, el formato de un nombre de fichero en Unix es: 


/directoriol/directorio2/ .. /fichero 


Cada uno de los componentes del nombre del fichero (direc- 
toriol, directorio2, fichero) pueden estar formados por cualquier 
secuencia de caracteres excepto por un slash “/”. Sólo son signi- 
ficativos los catorce primeros caracteres. No hay un límite para el 
número de directorios que se pueden especificar. 

El “mode” (modo) puede ser de tres tipos: 


e modo de acceso “indica que el fichero se va a abrir para 
lectura. Si el fichero no existe, se devolverá un error. 

e modo “w” especifica que el fichero se va a abrir para es- 
critura. Si el fichero ya existe no se tendrá en considera- 
ción el contenido anterior, perdiéndose por completo. Si 
el fichero no existiese, entonces lo creará la propia función 
fopen. 

e modo “a” especifica que el fichero se va a abrir para aña- 
dirle texto (append). Si el fichero ya existe, los nuevos da- 
tos se grabarán al final del mismo, y si nc existiese será 
creado por la función. 


En caso de que se produzca un error, fopen devuelve el va- 
lor NULL, definido en <stdio.h> como (char *)0. 
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E/S básica 


Las operaciones más sencillas que se pueden realizar sobre 
un fichero que ya esté abierto son getc y putc. 


int getc(fp) 
FILE *fp; 


gete devuelve el siguiente carácter del fichero especificado 
for fp, o EOF (definida en <stdio.h>). EOF se devuelve cuando se 
produce algún error o se alcanza el final de fichero. 
Análogamente: 


int putc(fp) 
FILE *fp; 


putc escribe el carácter dado en el fichero fp, y retorna “c” o EOF 
(en caso de error). 


feof(fp) 
FILE *fp; 


feof devuelve un valor distinto de cero si se ha llegado al final del 
fichero fp. 


ferror(fp) 
FILE *fp; 


ferror devuelve un valor distinto de cero si se ha encontrado al- 
gún error durante la lectura o escritura del fichero fp. * 


fclosc(fp) 
FILE *fp; 


íclose cierra el fichero fp, devolviendo un EOF si se produce al- 
gún error (por ejemplo, si el fichero fp no fue abierto previamente). 
La función freopen 


FILE *freopen(name, mode, fp) 
char *name; 

char *mode; 

FILE *fp; 


en primer lugar cierra el fichero especificado por fp y abre un nue- 
vo fichero, de modo que el fichero recién abierto reutiliza la es- 
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tructura FILE apuntada por fp. Freopen devuelve fp si la apertura 
se ha realizado sin problemas, o NULL en caso contrario. 


ungetc(c, fp) 
char Cc; 
FILE *fp; 


ungelc provoca que la siguiente llamada a getc(fd) (así como 
scanf, getw, gets, y las restantes funciones de lectura) lea el ca- 
rácter "c”. Mediante el empleo de ungete sólo se puede poner un 
carácter en el buffer del fichero fa. 


Ficheros estándar 


Cuando se inicia la ejecución de un programa € se produce 
la apertura automática de tres ficheros, antes de que llegue a pro- 
ducirse la llamada a main(). Estos ficheros son la entrada estándar 
"stdin” (standard input, normalmente el teclado), la salida estándar 
“sidout" (standard output, normalmente la pantalla de vídeo) y el 
fichero estándar de salida de errores “stderr" (standard error, nor- 
malmente la pantalla de video). 

Este encaminamiento normal de los ficheros estándar (panta- 
lla y teclado por defecto), puede verse modificado a través del 
intérprete de comandos de Unix y reencaminado hacia ficheros, 
“tuberías” (pipes) o regiones de memoria compartida. 

La librería estándar de E/S declara stdin, stdout y stderr como 
punteros a estructuras FILE para los ficheros de entrada estándar, 
salida estándar y fichero estándar de errores, respeclivamenle. 

Las funciones getchar y putchar son macros, en lugar de au- 
ténticas funciones, definidas en <stdio.h> como: 


ttdefine getchar() getc(stdin) 
Htdefine putchar(c) putc(c, stdout) 


Hay que advertir que stdin, stdout y stderr son constantes y 
no pueden ser reasignados mediante fopen como podría parecer 
natural, 

» 


stdout = fopen(“myfile”, “w”); : 


no es un procedimiento válido para redireccionar la salida están- 
dar al fichero myfile. Sin embargo, 


lreopen('myfile”, “w", stdout); 
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sí podría servir para este propósito, debido a que freopen cierra 
y abre el fichero, como comentamos anteriormente. 


Salida con formato 


printf es la función básica utilizada para salida con formato; es 
llamada en la forma: 


printf(control, argl, arg2, ..) 
char *control; 


Esta función acepta un número variable de argumentos, con- 
virtiéndolos, formateándolos e imprimiéndolos bajo las especifi 
caciones incluidas en "control”. Los caracteres del string de con- 
trol se vuelcan en el fichero de salida estándar, salvo el metaca- 


rácter "%", que se emplea como una especificación de formato pata 
controlar la impresión de los argumentos. 

Las es aciones de formato de salida comienzan con un 
%' y terminan con un carácter de conversión. Una es icación 


puede contener opcionalmente, descritos por orden: 


e un signo menos "-”, indicando que el siguiente argumento 
debe imprimirse justificado a la izquierda en su campo de 
salida; 

6 una cadena de dígitos, especificando el tamaño mínimo del 
campo de salida para ese argumento. Si el argumento ya 
formateado ocupa menos caracteres que la anchura del 
campo que le ha sido asignado se completará con carac- 
teres por la izquierda (o por la derecha si se ha especií- 
cado justificación a la izquierda). 


Si la especificación de la anchura del campo contiene un cero 
como primer carácter (por ejemplo 054), la longitud total del cam- 
po se completará con ceros en lugar de con espacios en blanco. 
La inclusión de este cero por la izquierda no debe confundirse en 
ningún caso con la expresión de una constante en octal (por ejem- 
plo, el carácter espacio en blanco, ASCII 32 en decimal, se expre- 
sa como 040 en octal). 

A diferencia de la salida con formato en Fortran, printf no trun- 
ca la conversión de argumentos en caso de que necesiten una ex- 
tensión mayor que la especificada para el campo de salida; 


e un punto, separando el ancho del campo de salida, de la 
precisión; 
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una cadena de dígitos que puede especificar tanto el nú- 
mero máximo de caracteres a imprimir de una cadena, 
como el número de dígitos a imprimir a la derecha del pun- 
to decimal cuando se están imprimiendo números en coma 
flotante (float o double); 


B 


una ele *l”, que indica que el siguiente argumento es un 
long it, en lugar de un int; 
algunas implementaciones de printf interpretan el carácter 


asterisco **” como una anchura de campo o precisión, con 
el significado de que el siguiente argumento indicará real- 
mente la anchura del campo o precisión del argumento a 
imprimir. Por ejemplo: 


printf("%.*s”, longitud, cadena); 


imprimirá al menos “longitud” caracteres de la cadena de 
caracteres “cadena”. 


Los caracteres de conversión de salida son: 


d 


e) 


y 


A 


ena 


- Convierte el argumento entero a un número decimal con 
signo. 

- Convierte su argumento (entero) a base octal. 

- El argumento entero se convierte a hexadecimal. 

- El argumento entero se convierte a notación decimal sin 
signo. 

- Interpreta su argumento como un único carácter. 

- Considera su argumento una cadena (string). 

- Convierte su argumento en coma flotante (interpretado 
como double, porque C siempre pasa los números en 
coma flotante como double) a notación exponencial, con 
signo, mantisa y exponente: 


[-]Jm.nnnnE[-]xx 


La longitud de la cadena de enes "n” viene determi- 
nada por la precisión, que es de € por defecto. (Recorde- 
mos que float proporcionaba 6 dígitos de precisión y dou- 
ble 15.) 


í - El argumento en coma flotante se expresa en notación de 
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coma fija: : : 
[-]mmm.nnnn 


La longitud de la cadena de enes “n” se toma nueva- 
mente de 6 por defecto. 


g - Convierte su argumento en cóma flotante a formato “%f" 
o “%e”, dependiendo de cuál de los dos proporcione una 
cadena de menor longitud. 


Cualquier otro carácter distinto de los especificados anterior- 
mente y que siga a un signo “%" se imprimirá literalmente. Así: 


printi('%4”); 


Imprimirá un único carácter “%. 
La aparición de los anteriores caracteres en mayúsculas (D, 


O, X, U,E, F o C) se interpreta como si una ale “1” precediese al 
carácter correspondiente en minúculas. Así 


printi("%D”, valor); 
es equivalente a 

printf(“%ld", valor); 

La salida con formato especificada por printf y enviada por 
esta función al fichero de salida estándar también se puede en- 
viar a otro fichero o a una cadena de caracteres, por medio de las 
funciones fprintf y sprintf, respectivamente. 

fprintf(fp, control, argl, arg2, ...) 

FILE *fp; 

char *cntrol; 
realiza su salida sobre el fichero especificado por fp. 

sprintf(string, control, argl, arg2, ..) 


lleva su salida al array string de acuerdo con las especificaciones 
de formato incluidas, en lugar de poner la salida en un fichero. 


Entrada con formato 
scanf es la función básica de entrada con formato. 


int scanf(control, argl, arg, ...) 
char *control; 


Esta función lee caracteres con formato a partir del fichero de 
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ae acuerdo a las espe 


> la Cac e COntrol. 

Cada uno de sus argumentos debe ser un puntero, especifi- 
cando la dirección de la variable donde deberá almacenarse el 
resultado. Obviamente, scanf es una función que necesita "llama- 
da por referencia”, pues va a modificar las variables que se le pa- 
san como argumentos. Un error frecuente en su uso consiste en 
la omisión de los “8” en cada uno de los argumentos. 

Los caracteres de separación como espacios en blanco, tabu- 
ladores y caracteres de salto de línea que figuren en la cadena 
de control, serán ignorados. 

Los restantes caracteres (excepto el '%”) se corresponden con 
una especificación de formato de entrada, pudiendo consistir en: 


e el carácter asterisco "*” indicando que el valor leído no 
debe almacenarse en la variable: 

una cadena de dígitos especificando la longitud máxima 
del campo de entrada (el número de caracteres a conver- 
tir); 

e carácter "n”, para indicar que el siguiente argumento debe 
ser considerado como un puntero a un short int, o el carác 
ter “1” indicando que debe ser tratado como un puntero a 


un long int, o a un double (equivalente a long float); 
e carácter de conversión. 


Las conversiones de caracteres de entrada permitidas son: 


d - Lee un entero decimal. El argumento debe ser un punte- 
ro a un entero (o un puntero a un short int o long int, si 
se hubiesen antepuesto "h” o 1" a la especificación de for- 
mato “d”), 

o - Lee un número entero octal. 

x - Lee un número entero hexadecimal. 

Cc - Lee un único carácter de la entrada. El argumento debe 
ser un puntero a un carácter. 

s - Acepta una cadena de caracteres terminada por un espa- 
cio en blanco o un carácter de salto de línea. El array de 
caracteres donde se guarde el resultado debe ser lo bas- 
tante grande como para contener todos los Caracteres y 
el carácter nulo terminador de cadena. 

e - Lee un número en coma flotante. El argumento debe ser 
un ficat (o un double, con especificación "le”). 

.! - Análogo al anterior. Acepta un número en coma flotante. 
| - Lee una cadena de caracteres limitada por separadores 
arbitrarios, almacenándola en el array apuntado por el ar- 
gumento. El corchete de apertura “[” va seguido por un 
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conjunto de caracteres y un corchete de cierre “]”. Si el 
carácter inmediatamente posterior a *[” no es un circun- 
flejo “6”, entonces scanf leerá solamente los caracteres 
que aparezcan en el conjunto de caracteres entre Cor- 
chetes. Por el contrario, si el carácter inmediatemente 
posterior a “[” es un circunflejo *”, entonces scanf leerá ca- 
racteres hasta que encuentre algún carácter de los con- 
tenidos en el conjunto de caracteres entre corchetes. 


Los caracteres de conversión d, o, x, e, y f pueden aparecer 
en mayúsculas o precedidos por una ele “1”, para indicar que el 
correspondiente puntero apunta a un long int o long float. 

scanf interrumpe la lectura cuando llega al final de la cadena 
de control, o cuando la entrada no cumple la especificación de for- 
mato impuesta (por ejemplo, intentar leer un literal con una espe- 
cificación %d de número entero). 

scanf devuelve un número entero indicando el número de 
asignaciones que ha realizado o OF si ha encontrado un final de 
fichero y esperaba leer más caracteres. 

Es importante aclarar que el valor cero no quiere decir que 
se haya alcanzado el final de fichero, sino que no se ha converti- 
do correctamente ningún campo de entrada. 

Análogamente a lo descrito para printf, scanf también permi- 
e que su entrada provenga de una cadena de caracteres O de un 
fichero. 


int fscanfífd, control, argl, arg2, ...) 
FILE *fd; 
char *control; 


es similar a scanf, con la diferencia de que toma su entrada del 
fichero fd, en lugar de "stdin”. 


int sscaní(control, argl,.arg2, ...) 
char *control; 


tomaría su entrada de una cadena de caracteres. Como la lectura 


de una cadena no viene acompañada de efectos laterales (side 
effects), sscanf es generalmente más útil que scanf o fscanf. 


E/S orientada a líneas 


La función fgets leerá la siguiente línea de entrada (incluido 
el carácter 1n) del fichero “fp”, guardando su entrada en el array 
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“line” que deberá estar terminado en un carácter nulo. Se leerán 
al menos “n-1” caracteres. fgets devuelve line, salvo que se haya 
alcanzado el final de fichero antes de llegar a leer ningún carácter. 


char *fgets(line, nchars, fp) 
char *line; 

int nchars; 

FILE *fp; 


La función fputs escribe la cadena de caracteres “line” sobre 
el fichero especificado por fp. 


fputs(line, fp) 
char *line; 
FILE *fp; 


La librería estándar también incluye las funciones gets y puts 
que leen y escriben de stdir. y stdout, respectivamente. Sin em- 
bargo, no son equivalentes a fgets y fputs dirigidas a stdin y stdout. 

La función gets 


char *gets(line) 
char *line; 


lee'la siguiente línea de entrada en stdin (incluyendo el 1n) sobre 
el array de caracteres “line”, que se supone tendrá suficiente es- 
pacio para contener la entrada, El carácter In no se almacena en 
“line”. A diferencia de fgets, gets devuelve “line” a menos que se 
alcance el final de fichero antes de que se haya podido leer nin- 
gún carácter. 

La función puts 


puts(line) 
Char *line; 


escribe la cadena de caracteres “line” sobre stdout seguido por 
un carácter n. 


Entrada/Salida binaria 


Las funciones básicas de E/S getc y pute son capaces de leer 
y escribir cualquier tipo de fichero (al menos bajo Unix), no sólo 
ficheros de texto. 

Se puede escribir una “palabra” (tipo de dato entero) sobre 
un fichero utilizando: 
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int putw(word, fp) 
int word; 
FILE *fp; 


putw devuelve la palabra escrita en el fichero, o EOF, si se ha 
producido algún error. 
Se puede leer una palabra de un fichero utilizando 


int getw(fp) 
FILE *fp; 


getw devuelve la palabra, o EOF, si se ha alcanzado el final 
de fichero antes de leer la totalidad de la palabra. Como EOF es 
un valor permitido para un int (pero no para un char), debería uti- 
lizarse feof para comprobar la cor.dición de final de fichero des- 
pués de getw. 

Cualquier variable de € puede escribirse como si se tratase 
de un array de caracteres. Por ejemplo, dada la declaración: 


union € 

lonmg int val; 

char clsizeof longd; 
73 545 


Leyendo o escribiendo el array de caracteres x.c se transfe- 
rirá exactamente el valor de x.val. 
La función general de lectura binaria es fread: 


int furitelpointer, size, number , fp) 
char *Xpointer; 

unsigned int size; 

unsigned int number; 

FILE Xfp: 


Esta función lee un número number” de objetos del fichero 
"fp", de un tamaño de “size” bytes cada uno de ellos, en el área 
de datos apuntada por “pointer”. fread devuelve el número de ob- 
ietos completos que ha podido leer. Así, cero equivale a un fin de 
fichero. 

La función general de escritura binaria es fwrite 


int fread(pointer, sizes, number», fp) 
char Xpointer; 

unsigned int sizez 

unsigned int number; 

FILE *fpx 


que escribe number” objetos del fichero “fp”, de “size” bytes cada 
uno de ellos, en el área apuntada por “pointer”. fwrite devuelve el 
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número de objetos completos escritos, que coincidirá con num- 
ber a menos que se haya producido agún error. 
Un ejemplo típico de utilización sería: 


long int x3 
FILE *X*from, *Xto; 


while (fread((char *X)%x, sizeof x, 1, fron) == 1) 
furite((char *X)kx, sizeof x, 1, to); 


copiando un long del fichero “from! al fichero “to”. 

Es importante observar el empleo del cast 8:x para obtener 
un char *. La expresión £(char *)x que fuerza a que “x” sea del 
tipo char, no funcionaría porque (char)x no es una variable, y no 
se puede aplicar un €, 

Aunque la utilización de fread y fwrite es portable, los fiche- 
ros manejados por ellos no tienen porqué serlo, pues estos fiche- 
ros son fiel reflejo del tamaño y formato de los objetos de datos 
representados por el ordenador en cuestión. 

Si se necesita escribir ficheros portables es mejor emplear 
funciones de desplazamiento y máscaras de bits o utilizar datos 
con formato. 


Acceso aleatorio a ficheros 


La E/S de ficheros en lenguaje C es habitualmente secuen- 
cial. Sin embargo, un fichero puede ser leído o escrito en cual- 
quier orden. 

La función (seek 


fseek(fp, offset, origin) 
FILE *fp; 

long int offset; 

int origin: 


obliga a que la siguiente llamada a gete o putc tenga lugar sobre 
la posición offset del fichero. El origen es 0, 1 6 2 indicando si el 
offset es relativo al principio del fichero (0), a la posición actual 
(1) o al final del fichero (2). Ñ 

La función 


long int ftell(fp) 
FILE *fp . 


devuelve la posición actual del offset a partir del principio o del 
final del fichero fp. 
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La función 


rewind(fp) 
FILE *fp 


es equivalente a: 


fseek(fp, OL, 0); 


Entrada/Salida con buffer 


Las llamadas al núcleo del sistema (system calls) para reali 
zar operaciones de E/S ocupan una gran cantidad de tiempo, com- 
paradas con una llamada a tunción. Una llamada al núcleo consis- 
te en utilizar funciones del núcleo del sistema operativo de una 
forma semejante a como se manejan las funciones “normales” en 
un programa. 

Para realizar una llamada al núcleo del sistema se requiere 
un trabajo extra, comparándolo con una llamada a función, nece- 
s'tándose almacenar información adicional durante la llamada y 
el retorno, además de la sobrecarga necesaria para encontrar un 
manejador de la llamada al núcleo, traspasar argumentos, etc. 

La sobrecarga en las llamadas al núcleo Se puede disminuir 
mediante la "bufferización”: se puede transferir un único carácter 
acia un buffer desde el proceso de usuario; una vez que el 
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buffer se ha llenado (para ficheros de salida) O vaciado (para fi- 
cheros de entrada) se puede transferir la totalidad del buffer lleno 
de caracteres, 

Este procedimiento reduce el tiempo de ejecución de un pro- 
grama, pero puede tener otros efectos. En particular, si putchar 
pone caracteres en un buffer, estos caracteres no aparecerán en 
la pantalla a medida que vayan siendo enviados por putchar; es 
más, ni siquiera lo harán después de que se haya producido el re- 
torno de putchar, con la molestia de que no aparece eco inmedia- 
to en pantalla. 

Si un programa termina anormalmente por cualquier causa, la 
salida que tenía bufferizada se quedará sin enviar al fichero de 
salida. 

Del mismo modo, los errores que se estén produciendo en la 
escritura no aparecerán hasta que se hayan escrito los datos del 
buffer, lo cual podría suceder bastante después de que la llamada 
a putc haya enviado los datos. 

La "bufferización” de entrada causa generalmente pocos pro- 
blemas, salvo en el caso particular de que más de un proceso esté 
leyendo simultáneamente el mismo fichero. 
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Realmente no hace demasiada falta controlar la bufferización 
de los datos; dejándolo en manos de la librería estándar de E/S 
podemos confiar en que la librería “hará las cosas bien” sin nece- 
sidad de control (por ejemplo no bufferizará los datos de salida a 
un terminal). ; 

No obstante, la librería estándar de E/S posee funciones que 
permiten validar c invalidar la "bufferización” sobre un fichero, así 
como una función que fuerza a la librería de E/S a que envíe de 
inmediato los datos retenidos en el buffer. 

La función setbuf E 

setbuf(fp, buffer) 

FILE *fp; 

char *buffer; 


informa a la librería estándar de E/S de que utilice el array buffer 
de BUFSIZ caracteres (definido en la <stdio.h>) para bufferización. 
Si buffer es igual a NULL entonces el fichero no estará bufferizado. 
setbuf debería llamarse antes de que se vaya a realizar cualquier 
operación de E/S sobre el fichero. 

La función 


fiush(fp) 
FILE *fp; 


fuerza que cualquier dato bufferizado del fichero fp sea escrito en 
el fichero. fflush devuelve EOF para indicar un error (por ejemplo, 
la imposibilidad de grabar los datos en el fichero) o cero en el 
caso contrario. 


EFICIENCIA Y PORTABILIDAD EN El C 


a capacidad de escribir programas que utilicen 
con eficiencia los recursos del sistema, estén li- 
bres de errores (error-free) y sean fácilmenle 
transportables a otros computadores son los sig- 
nos que indican el paso de un buen programa- 
dor y el uso de un lenguaje adecuado. 


Eficiencia 


Cuando hablamos de eficiencia en un programa de ordena- 
dor nos referimos tanto a Su velocidad de ejecución como a la uti- 
lización óptima de los recursos críticos del sistema (memoria y 
ocupación en disco, básicamente). No basta con que los progra- 
mas funcionen, sino que además deben hacerlo de manera que su- 
pongan la menor carga posible para el ordenador y no gasten sus 
recursos inútilmente. 

Normalmente, la optimización de un aspecto del programa 
suele degradar otros. Por ejemplo, hacer que un programa se eje- 
cute más rápido normalmente requiere que se duplique el códi- 
go fuente en algunos Casos, en lugar de realizar llamadas a fun- 
ciones. También se puede conseguir un ahorro de ocupación en 
disco, utilizando empaquetado de datos, a costa de degradar la ve- 
locidad de acceso a los datos. Estas y otras posibilidades de me- 
jora de la eficiencia pueden ser frustrantes para los no-programa- 
dores, a quienes les puede costar comprender cómo la mejora de 
una característica de un programa puede afectar negativamente 
a otras, llegando incluso a empeorar el resultado final, 
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Afortunadamente, existen algunas técnicas de programación 
que mejoran siempre la eficiencia en cualquier sisterna que con- 
sideremos. 

Por ejemplo, el uso de operadores de incremento ++ y decre- 
mento -- es más rápido y compacto que la utilización de opera- 
ciones como X = x + 1 (frente a x++). 

La utilización de funciones puede resultar perjudicial en oca- 
siones, como es en el caso de una función que sea llamada desde 
el interior de un bucle un gran número de veces. Pensemos en 
el siguiente ejemplo: 


a 


/* Funcién que evalúa el valor del polinomio 
Rx + 3% —1 en el intervalo de valores de O a 100 
*x/ 


for (i = 03 i < 100; i++) € 
x= polin((double) i); 
prints" d Fry ds. 04 


double pulintz) 
double 2; 
í 
return ((2 + 3) kz —-1)5 
> 


Sería más eficiente la escritura del programa en la forma: 


for (i=0 i< 1005 i++) € 

= (double) i; 

E (RR +43) Yo -L)3 
a NE LT 
3 


donde hemos eliminado la llamada a la función polin, que debería 
realizarse 100 veces, con lo que esto sigrifica en cuanto a secuen- 
cias de llamada y retorno de función y manejo de stack: la llama- 
da requiere salvar el estado de determinados registros de la CPU 
mediante la instrucción push de ensamblador, y el retorno requie- 
re la recuperación de sus valores originales mediante la instruc- 
ción pop; en ocasiones la función exige además la utilización de 
variables automáticas que han de ser definidas en cada llamada 
a función. 

La utilización de punteros en lugar de índices de arrays ge- 
nera un código más compacto en todos los casos, si tenemos en 
cuenta que internamente el compilador convierte todas las refe- 
rencias a subíndices en manejo de punteros. Por ejemplo, 


for (j3 = D35 £ 
io+= alj++1; 
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“accedería al elemento 'j” del array creando internamente un pun- 
tero al elemento O del array y desplazando el puntero en “5” ele- 
mentos, incrementando posteriormente 'j" en 1. Por el contrario: 


:S 
ji = Ktp++); 


23 


Sería más eficiente porque "p” ya es directamente un puntero 
al elemento 'j" del array al que deseamos acceder, El manejo de 
punteros es muy utilizado en los casos en que se desea acceder 
secuencialmente a todos los elementos de un array. 

En algunos círculos C es considerado como un lenguaje críp- 
tico, difícil de leer y de escribir. Esta mala reputación es debida 
integramente al estilo de algunos programadores, amigos de es- 
cribir programas difíciles de comprender incluso para ellos mis- 
mos cuando ha transcurrido algún tiempo desde su escritura, de- 
bido a la utilización de nombres de variables inadecuados, pro- 
blemas muy complejos llenos de “trucos” y utilización de expre- 
siones complicadas en la misma línea sin hacer uso de variables 
intermedias. Todo esto puede hacer en ocasiones que los progra- 
mas sean un poco más rápidos, al precio de hacerlos ininteligi- 
bles, siendo más razonable plantearse de antemano la idea de si 
no sería posible mejorar el algoritmo empleado, reducir el núme- 
ro de llamadas a funciones o, en casos muy extremos, codificar al- 
guna subrutina en lenguaje ensamblador, aunque esto requiere 
una muy buena razón para hacerlo. 


Portabilidad 


La portabilidad en lenguaje C se refiere a la posibilidad de 
transportar programas de uno a otro ordenador y ponerlos en fun- 
cionamiento ólo volver a compilarlos. La portabilidad se re- 

álo a los programas fuente; no tiene sentido alguno intentar 
transportar programas compilados de uno a otro ordenador, pues 
han sido compilados haciendo uso de las facilidades suministra- 
das por un determinado hardware, que no tiene porqué estar pre- 
sente sobre otro, salvo que expresamente nos estemos refiriendo 
a ordenadores compatibles y funcionando bajo un mismo sistema 
operativo. 

Posibles causas de pérdida de portabilidad son la utilización 
de "números mágicos” o constantes dependientes del ordenador 
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considerado, como pueden ser el tamaño en bytes de los buffers 
del sistema, la utilización de secuencias de escape y control par- 
ticulares para determinados periféricos y el empleo de funciones 
específicas de un sistema operativo que no tienen por qué estar 
presentes o funcionar igual en otros sistemas 

Para evitar el empleo indiscriminado de "números mágicos” 
es conveniente el empleo de las instrucciones define e Hifdef del 
preprocesador de lenguaje C, de manera que permita la modifi- 
cación de determinadas constantes antes de proceder a la com- 
pilación del programa. Por ejemplo: 


define UNIX 1 
*define ECOS 2 
Hdefine MS_DOS 3 


" 


flag = UNIX; 


+Hi4 flag == UNIX 
OFÁBELrOR propios del S.O. UNIX 
Hendif 
Hi4 flag == MS_DOS 
Parámetros propios del S.0. MS_DOS 
" ' 


endif 


De este modo podríamos emplear parámetros y funciones 
propios de cada sistema operativo con la seguridad de que los 
programas serían independientes del sistema considerado. 

Una causa posible (y peligrosa) de pérdida de portabilidad 
es la redefinición de nombres de funciones, como los contenidos 
en la librería de C estándar. Si redefinimos funciones como printf, 
strcpy, strlen, etc., por otras funciones propias que teniendo el mis- 
mo nombre manejen distintos argumentos de llamada y retomo, 
tendríamos problemas de funcionamiento obvios. 

Otra causa de pérdida de portabilidad es la utilización de se- 
cuencias de escape y control específicas para determinados pe- 
riféricos, como pantallas e impresoras de una determinada marca. 
Si nuestro programa ha de funcionar algún día con otros periféri- 
cos tendremos serios problemas de funcionamiento. La adopción 
de las secuencias de escape y control apropiadas para cada pe- 
riférico ha de efectuarse en tiempo de ejecución del programa, 
en lugar de en tiempo de compilación. Lo más normal es proce- 
der a la lectura de un fichero de parámetros conteniendo las ca- 
racterísticas de cada periférico considerado. O:ra posibilidad (me- 
jor) es la utilización de funciones de librería que hagan uso de 
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esta posibilidad (lectura de un fichero de parámetros) y además 
utilicen algún tipo de optimización, como el empleo de determi- 
nadas funciones que no están presentes en todos los termina- 
les (borrado e inserción de líneas, por ejemplo), o manejen por 
software otras posibilidades, como ventanas de texto. Estas posi- 
bilidades y otras son manejadas por un paquete estándar como 
es el curses de la Universidad de Berkeley, existente en la mayo- 
ría de las instalaciones que trabajan con lenguaje C. 
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UN EJEMPLO COMPLETO: EL PROGRAMA “calles.c' 


modo de glosario de lo visto hasta ahora vamos 
a desarrollar un programa que maneje muchos 
aspectos interesantes del lenguaje C, presentes 
en la mayoría de los programas que vayamos a 
realizar. 

El programa 'calles.c' tiene como misión mo- 
dificar el nombre de una calle (que se pasará 
como argumento de entrada) realizando sobre 
RTS él una serie de abreviaturas (por ejemplo, cam- 
biar teniente' por 'tte' o 'señor' por 'sr”) y eliminando palabras ro 
deseadas ('de' 'el', la”, ..). Se podría utilizar para minimizar el efec- 
to de posibles duplicaciones en una base de datos que contuvie- 
se nombres de calles, con entradas como: 


“SANTA CATALINA, PARQUE DE” 
"SAN ANDRES, CALLE” 


El programa devolvería como salida los nombres: 


"STA CATALINA PQE” 
"S ANDRES CALLE” 


donde ha reducido las palabras 'SANTA' y 'SAN cambiándolas por 
'STA' y 'S, y eliminado la partícula 'DE' y las comas de separación. 

Pensemos ahora en una gran base de datos, con 50.000 o más 
nombres de calles, que constituyan un índice, y el efecto de en- 
tradas como: : 


"SAN ANDRES, CALLE” 
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“S. ANDRES CALLE DE” 
"DE SAN ANDRES, CALLE” 
“SAN ANDRES CALLE” 


Nuestro programa evitaría esta dispersión de nombres y los 
transformaría en el único nombre: 


"S ANDRES CALLE” 


facilitando su localización y eliminando la posibilidad de claves 
duplicadas por error. 

El ejemplo no es muy importante en sí mismo, pero maneja 
conceptos aislados que sí son de interés; argumentos en la línea 
de llamada a un programa, manejo de arrays y estructuras con 
punteros, funciones propias y de la libreria C estándar, y estruc- 
turas de control muy empleadas en C (for, if, if else, switch). 

Para compilar el programa calles.c' habremos de utilizar el co- 
mando de sistema operativo: 


cc -o calles calles.c 
y ejecutar el programa como: 

calles 'SAN ANDRES, CALLE DE' 
obteniendo como resultado: 


Entrada: ->SAN ANDRES, CALLE DE <- 
Salida: ->S ANDRES CALLE<- 


El listado del programa, casi sin ningún tipo de comentario 
para facilitar su seguimiento y el de su estructura, es el que sigue 
(al final del capítulo se incluye con todos los comentarios autoex- 
plicativos): 


tinclude <stdio.h> 


char callel3013, palabral301; 
char *X tabla(); 


struct mapa í 
char *Xcar_in; 
char Xcar_out; 
3 


struct mapa calles[1 = £ 
"ALCALDE", “ALE”. 
"ALFEREZ"”, "ALFZ", 
"ALMIRANTE", "ALMTE”, 
"ARQUITECTO, " "ARQTO" 


"DE", tchar X)0, 


“ELY, (char X)0, 
"NUESTRA", "RA", 
"PARQUE", "PQE", 
"PRESIDENTE", — "PDTE”, 
"PUERTO", “PTO", 
"REPUBLICA", "RPECA", 
"SAN", 5, 
"SANTA", STA", 
"SANTO", STO 
"SEÑOR “: A " SR ” » 
"SEÑORA", "SRA", 
“TENIENTE”, "TE", 
"Y iRGEN", "ya" 
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short hi_calle = sizeof (mapa) / sizeof (struct calles); 


/* Programa principal *X/ 


mainlargc, argv) 
int aroc; 
char *XargvCl; 
i 
unsigned 13 
char *Xp; 


for (i =0, p = argvl1l;3 Xp p++) t 
switch (Xp) £ 


case "3 /k separadores de X/ 
case *.?: /K palabra x/ 
case *,”: 
palabratlil = *M0*; 
strcat(calle, tablalpalabra)); 
strcatícalle, " "); 
1i= 0; 
break; 
default: 


palabrali++] = Xp; 
7 


z 

palabralil = *M0”3 ' 
strcat(calle, tablalpalabra)); 

strcatícalle, " "); 

trimarícalle); 

printf ("AnEntrada: —“>%s<-", argvi1J); 

printf ("AnSalida : ->síAin", calle); 


1 


/* Función tablals) */ 


char * tabla(s) 
char s[1; /* argumento de llamada a *tabla” x/ 
£ 


struct calles Xp; 

int low, high, mid, cond; 
low = 0; 

high = hi_calle; 


while (low <= high) £ q 


mid = (low + high) / 2; 
p= tmapalmidl; 
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if ((cond = strcmp(s, p->car_:m)) < 0) 
high = mid — 13 
else if (cond > 0) 
low = mid + 1; 
else 
/* ¡Encontrado! */ return ((char *) p->car_out); 
7 
return ((tchar X) s); 
> 


/* Función trimar(s) *k/ 


void trimar(s) 
char *s; 
t 
char *Xp; 
unsigned dbl = 0; 


for (p = strj Xp == ? '; p++) 


; 
while (Xp) £ a 

if (kp == * *) € 
db1l = 1; 
p+r+; 
continue; 

5 

if (db1) £ 
E E 

, dbl = 0; 

3 

Xs+t+ = kp++; 


*Xs = "10? 


Al principio del programa se declaran las variables globales 
calle[30] y palabra[30], como cadenas de caracteres, y se declara 
tabla como una función que devuelve un puntero a una cadena 
de caracteres. 

Luego se define la estructura mapa, compuesta por car_in y 
car_out, punteros a las cadenas de caracteres de entrada y salida, 
respectivamente. calles se define como una estructura mapa, y se 
procede a inicializarla. 

El número de elementos que contiene la estructura calles se 
calcula como 


short hi_calle=sizeof(calles) / sizeof(slruct mapa); 


Es decir, el número de elementos (hi_calle) equivale al tama- 
ño (sizeof) de la estructura calles, dividido por el tamaño de un 
elemento (mapa). 

En este caso main aparece como main(argc, argv), debida a 
que va a manejar argumentos en la línea de llamada (el nombre 
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de la calle a modificar). El argumento se pasa entre comillas sim- 
ples "'”, para evitar que los espacios en blanco intercalados ha- 
gan que sea considerado como varios argumentos. 

Mediante un bucle for y el desplazamiento de un puntero 
arg[1], se va utilizando el argumento de entrada carácter a carác- 
ter. Si se encuentra un carácter distinto del separador (los sepa- 
radores son '”, '. y '/), se carga en el array "palabra" y se incremen- 
ta el subíndice “1”. 

Si, por el contrario, se alcanza un separador entonces se aña- 
de un carácter nulo a «palabra», como terminador de cadena de 
caracteres, y se envía como argumento a la función tabla, que nos 
devolverá un puntero a una cadena, bien apuntando a una cade- 
na distinta de la de entrada, si la encuentra en la estructura calles, 
o la misma cadena de entrada si no la ha encontrado, Al final le 
añade un espacio en blanco como separador, y reinicializa el su- 
bíndice “¡” a O, para preparar la búsqueda de otra palabra. 

Esta secuencia se repite hasta alcanzar el final de la cadena 
de entrada argv[1]. 

La función tabla tiene como misión efectuar la búsqueda de 
una palabra en la estructura calles, devolviendo la palabra modi- 
ficada en caso de encontrarla. La búsqueda en la estructura se im- 
plementa mediante una búsqueda binaria. 

Se compara la cadena a buscar con la correspondiente a la 
mitad de la tabla (que ha de estar ordenada), si es mayor la ca- 
dena buscada se repite la búsqueda en la mitad superior de la ta- 
bla, si no en la inferior; y así sucesivamente hasta encontrar el ele- 
mento buscado o finalizar con una comparación entre dos posi- 
bles elementos. Si al final del proceso no se ha encontrado la ca- 
dena buscada, se devuelve un puntero a la misma cadena, para 
que el programa principal la añada a la cadena obtenida como re- 
sultado de salida. ' 

Al final, se llama a la función trimar para que elimine los es- 
pacios en blanco que pudiera haber al principio, final, y embebi- 
dos en la cadena obtenida como resultado del programa y se im- 
primen para su comparación las cadenas de entrada y salida. 

La función trimar elimina los caracteres en blanco que pudie- 
ra haber en una cadena de caracteres, mediante el desplazamien- 
to de un puntero. Cuando encuentra un blanco intermedio, pone 
a uno un indicador (dbl) y continúa la exploración de la cadena 
sin copiar ningún carácter a su salida. Cuando se encuentre un ca- 
rácter distinto de blanco, si tiene a 1 el indicador dbl, entonces 
pone un blanco a su salida, y reinicializa dbl a 0. Luego copia so- 
bre su salida el carácter distinto de blanco que acababa de en- 
contrar. De este modo, ha convertido la aparición de múl'iples 
blancos en uno solo. Al finalizar la exploración de la cadena, sitúa 
un carácter nulo como terminador de cadena de caracteres, 
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Para finalizar vamos a incluir el listado completo del progra- 
ma, con todos sus comentarios autoexplicativos. 


MOIOIOROIOIOO OOOO OOOO aaa lojolalaldalaalo lolo ajolololoiioKK 


* Define los arrays de caracteres *calle” y *palabra” * 
* con capacidad para 30 caracteres, y '"tabla” como * 
X una función que retorna un puntero a un char * 
MOROROIOROIOROIOOOOO OOOO OOOO OOOO IOOIOIOGIOIOOIOIOIoOK 
*x/ 


char caliet301, palabral303; 
char * tabla(); 


PARAR OROROROROOOROROOIOOOOIOOIOOOIOOOOOIOOIOOOOCOIOOIOOIOIOIOIOOICIODO k 
* Define una estructura "mapa? compuesta por dos * 
X punteros a un char, denominados *car_in” y *car_out?”. x 
*X *"car_in” es la palabra de entrada que debe buscarse, * 
* y *car_out” es la palabra que debe devolverse a la * 
*X salida, si se encuentra "car_inm” x 
IIOOIOROROIO OO IOIOIOOIOOOIOOIO OOOO PISO loa ojala joiaidlodidlojook 
x/ 

struct mapa £ 

char Xcar_inz 
char *Xcar_out; 


e 
5 


DIRIJO PIOOIOIOOIOIOIOIOO OOOO IOOOIOIOOOOIOIOIOICOIOOIOOIOIOOOIOOIOIOIBIGIOROJOjOK: 


Xx Define *calles” como un array de estructuras 'mapa”, k 
* y lo inicializa a un conjunto de valores especificado x 
* correspondientes a *"palabra-de-entrada” y 'palabra-de xk 
* salida”, kx 
III ROO OOO OOOO OOOO OOOO IO OIOIOIOOIOIOOIOOIOIOIOION: E 
*/ 
struct mapa calles[1 = € 

"ALCALDE", "ALC", 

"ALDEREZ", ALEZS 

"ALMIRANTE", YALMTE"”, 

"ARQUITECTO, " "AROTO" 

De tchar X)0, 

E (char X)0, 

"NUESTRA", "NRA", 

"PARQUE", "POE", 

"PRESIDENTE", "PDTE”, 

"PUERTO", "PTOS 

"REPUBLICA", "RPFECA”, 

"SAN", gn, 

“SANTA”, "STA”, 

"SANTO", "STO", 

"SEROR”, "se", 

"SERORA", + "SRA", 

"TENIENTE", "TES 

"VIRGEN", "yg" 


DIARIO OOOO ROO OOIOIOOIOOIOOIOOOIOOIOIOIOIOOIO OOOO OIGO okok 
hi_calle es el numero de elementos de la estruct.calles *X 
calculado como el tamaño en memoria de la estructura * 
dividido por el tamaño (sizeof) de un elemento de la A 
estructura. De este modo para añadir nuevos elementos k 
basta con incluirlos en su lugar en la estructura, 

el programa calculará automáticamente el número * 
de elementos 
PORRO ORO OOOO OOOO OOOO OOOO OOOO OOOO KK 


* 
* 


Er A e 


hurt hi_calle = sizeofímapa) / sizeof (struct calles); 


10% 


aaa Sao laaala 1 aaiaablaaldla dada aa daa oaOadoIdoota 


xXx Frograma principal. Notemos que *main” va a manejar * 
* argumentos de llamada, por eso escribimos * 
* mainíaroc, argv) y dafinimos argoc y ara * 
Aaa ada Ral 910IblR9 9991109100100 0 0DbJok 
x/ 
maintargc, arav) 
int argc; /* define los argumentos de main */ 


char *karovE]; 


z 


unsigned i; /k define variables locales a main x/ 
char *Xp;3 
OOOO RRE Ia da DDRIORaDnono 
* Explora el argumento argvi11 mediante el puntero 2, k 
X considerando los caracteres ” a e NI ARO x 
X separadores de palabras k 
Aaa ROO RP RadÓ idad dlaodl9 oa Rla oido 0dBODIODb 
x/ 


for (i = 0, p = árovitld; Xp p++) t 
switch (kp) € 


case /*X separadores de */ 
case ?. /Kk palabra x/ 
case *».,*: 


RADO RR RRR0 dada Rada Ral oR olaa DRODDRBIOdidbobERR 


* Pone un caracter nulo "»M0” como terminador de strinma * 
* en la posición i del string palabra * 
A ORO addon haa R8 dada oa aa RRRBaDDbnd 
*x/ 


palabralil = "N0*%; 


EEE ETE DATO 503550 00 A 
Concatena con *calle” (resultado del programa) el valor * 
devuelto por la función tabla”, que retorna un puntero *k 
a un char, pudiendo ser tanto una palabra modificada xk 
(si se ha encontrado en la estructura *mapa”), como la x 
misma palabra a buscar, en caso de que no haya sido * 

x 

* 


A HA 


encontrada en la búsqueda 

aa ao aaa Rda alada aRalO alo Rla ORD BBOnEK 
x/ 
strcatícalle, tabla(palabra))1 
is=00 t 

breaks 


E A 


X No separador: carga el caracter apuntado por p en la xk 
* posición i del array *palabra”, incrementando luego * 
Xk la variable i, preparándola para la siguiente búsqueca XK 
AA RPaaldR0 dada daRR ado d o ada RRIODRDRRRROÓbIaDoR 
xk! 

default: 


palabrali++J = %p; 


ARO oRRROoRRRaRRRRRRR 0 RRoD oRaaRaDDaDa0ODDDIBObuna 


xXx Termina el string palabra, Y realiza la última x 
Xx búsqueda en la tabla de equivalencias * 
aaa aaa laa dado Bo oa ooo oRoo4oJoX 
*/ x 


nalabratil = *M0”; 
strcat(cálle, tablalpalabra)); 


: 103 


104 


ROO RO OOOO OOOO OOOO OOOO OOOOIOIGIOIGIOIOOIOOO« 


* Elimina (trima) los blancos múltiples que pueda * 
X* contener calle y muestra el nombre de entrada * 
*X y el obterido como resultado del programa * 


PAJARO OOOO OOOO OOOO OOOO OO OOOO OOOO Odo 
x/ 

trimer (calle); 

print+("InEntrada: —>%sí-", argv[11); 


printf("AnSalida + -2sítAn", calle); 
» 
Po 
IRA ROO OOO OOOO OOO OOOO OOOO OOOO OOOO IPR OlOldRookX 
Xx Función *"tabla”. que retorna un puntero a un char k 


IO IO OOOO GIO Olla lcio o dioior 
*x/ 

char *X tablaís) 

char s[1; /* argumento de llamada a 'tabla” */ 


' 
t 


PARAR OOOO oO Odo lRaIdioodook 


* Define variables automáticas (sólo válidas en el * 
X interior del cuerpo de la tunción 'tabla'): * 
MORO IONO ORIO OOOO OO: 
x/ 


struct calles Xp; 

int low, hioh, mid, cond; 
low = 0; 

high = hi_calle; 


MO OOOO olla dalla lalo lalo lplojdloldlodolook Xx 
Búsqueda de "palabra? en la tabla "mapa”. Se emplea * 
uma búsqueda binaria (bimary search) por razones de k 
eficiencia, para disminuir el número de comparaciones A 
a realizar. La búsqueda biraria consiste en explorar k 
una tabla de nombres dispuestos en orden creciente, * 
de manera que en cada momerto se compara sólo con el x* 
elemento central de la tabla. Si el elemento a buscar * 
es > que el elemento central, entonces se seguirá bus- * 
cando en la mitad superior de la tabla, y, sinó en la * 
inferior. * 
MOJO OOOO ORAR IO IO OOOO look 
x/ 


E 


while (low ¿<= high) £ 
mid = (low + high) / 2; 


IOIOOGOOOIOOOOOGIOIOSOOIO SOS lll SII lO GO llddIadIoGOjooOKk 
*X Inicializa el puntero p para que apunte a la dirección * 
* del elemento central de la tabla * 
MOSAICO OOOO OOOO IO OIDO OOOO 

p = kmapalmidl; 


POROROIOROIOROOIORO ORIO ORO AO AA AR AA RRA ARA RARA AAA AAA KARA 
kX Realiza la comparación, mediante la función de la k 
*X librería C estándar *'strcmp”, que compara dos strings x 
* entre si, retornando un núrero > O si el primero es * 
* mayor que el segundo, = O si son iguales, y < O si * 
kx el primero es menor que el segundo * 
IOOIOOIOOIOROIORIOROIOOO O OOOO OOOO OOOO OOOO 
x/ 

if ((cond = strcmp(s, p->car_in)) <.0) 
high = mid -— 1; 


else if (cond > 0) 
low = mid + 1; 
else 
/% ¡Encontrado! */ return ((char *) p->car_out); 
7 


/X No encontrado al final de la búsqueda *X/ 


return ((char *X) Ss); 


La 


RARO ROAAORORRRRRAORRRRRDRDRRRRR a RDRÓDRandRdRRRODOODODEDRE 
Xx Función *"trimar”, que elimina los espacios en blanco *x 
*X al principio y final de un strings, asi como los espa- * 
* cios en blanco múltiples intercalados. Ante una entrada X*X 

x 
xk 
* 


*X como: e ab ¡A 
kx retornaríia: "ab e de f* 
olaa Roda adaRRRORaRRR bbdd bRaODRDOÓRDao 
x/ 
void trimar(s) 
char X=; 
tí 


char *Xp; 
unsigned dbl = 0: 


ARaORORRadaaRnaRRRRDRa nadaa addon o RaODRIDRRDDE 
* Elimina los espacios en blanco al principio del string x 
FARRO RO 0laR daa dla aRRa dao bado dano DRRDDDEE 
*/ 

for (p = strií tp == * *5 p++) 


a ooaRaataRa ala aaa dRRdlaO Daba Dado bidoODRIÓdDDDR 
*x Explora el resto del string, poniendo el indicador * 
* *"db1* a 1 cuando encuentra un espacio en blanco, Y x 
xXx no copiando los caracteres a la salida. Cuando * 
* encuentra un caracter distinto de blanco, pone * 
* un espacio en blanco a la salida (si habia blancos * 

k máltiples) y copia los caracteres a la salida ' * 

OOOO 0100821021920 01909010010 919010913001010 80101000 0J010010% 

x/ 

while (Xp) 


/* Encuentra un espacio en blanco Ni 
if (kp == * *) € 
db1 = 1: 
pt; 
continue; 


> 


AO RR RRRDIO RR DRIRd Dad ada RoORRaODODOPRROLA Es 
* Caracter distinto de espacio en blanco. Copia x 
* blanco sobre la salida, pone a Cero el indicador * 
* *db1*? y copia un caracter sobre la salida, k 
*X desplazando los punteros de entrada y salida * 
ORAR ORR aaa Ro abad baaa DdaalRaoaaaDbBioaR 
*x/ E 

if (db1) s 
kstr = "73 
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MAS O OOOO aa lla lololaiojolak 


* Pone un caracter nulo *"X0” como terminador de string *x 
IGG III SII ollo lololokXK 
x/ 
Xs = *"X0”*; 
> 
e 


LA LIBRERIA C ESTANDAR 


a librería € estándar consiste en varios tipos de 
funciones para control de E/S, tratamiento de ca- 
denas y caracteres, y funciones de tiempo y fe- 
cha. Todas las funciones son incluidas automáti- 
camente por el compilador de lenguaje C, sin 
necesidad de utilizar opciones especiales de 
compilación. 

La mayoría de las funciones requieren la uti- 
lización de algún fichero de Htinclude, que de- 
bería incluirse al principio del (primer) fichero a compilar. 


ABS - esta función, aplicada sobre una variable literal, nos devuel- 
ve su valor absoluto. 


int abs(i) ! 
int i; 


ATOF - convierte una cadena de caracteres que contenga valores 
numéricos en su valor numérico expresado en doble precisión. 


double atof (nptr) 
char Anptrs 


BSEARCH - búsqueda binaria. 


char *bsearch(í (char *)key, (char *X)base), nel, 
sizeof (Xkey), compar) 

unsigned nel; 

int (kX compar) (); 


CEIL - ante un valor numérico con decimales conserva la parte en- 
tera, despreciando los decimales. 
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tinclude <math.h> 
double ceil (x) 
double x3 


CLOCK - devuelve el tiempo de CPU utilizado 
long clock (t); 


CONV - traslada caracteres. 
+tinclude <ctype.h> 


int toupper (c) 
int cs 
(a) 
tinclude <ctype.h> 
int tolower (c); 
int: es 
0) 


*tinclude <ctype.h> 
int toasciiíc) 
int e; 


CRYPT - genera encriptación por clave DES 


char X*Xcryptíkey, salt) 
char *Ykey, Xsalt; 


void settkey (key) 
char *Xkey; 


void encryptíblock, edflag) 
char Xblock; 
int edflag; 


CTERMID - asocia un nombre de fichero al terminal conectado. 


Hinclude <stdio.h> 
char *Xctermid(s); 
char *Xs; 


CTIME, LOCALTIME, GMTIME, ASCTIME, TIMEZONE - convierte 
la representación interna de la fecha y hora del sistema a un for- 
mato alfanumérico. 

Htinclude <time.h> 

char *ctimelclock); , 

long *clock; 


*include <time.h> 
struct tm Xlocaltimeífclock); 
long *clock; 
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Htinclude <time.h> 
struct tm *komtimelclock); 
long *clocks 


o 
tinclude <time.h> 
char *asctimettm); 
struct tm *Xtm; 
O 
Htinclude <time.h> 
extern long timezone;z 
o 
Htinclude <time.h> 
extern char *Xtzname; 
o) 


Htinclude <time.h> 
void tzsc() 


CTYPE - ordena los códigos ASCII correspondientes a valores en- 
teros según una tabla. 
Hinclude <ctype-»h> 
int isalphalc); 
¿int c; 


CURSES - funciones de manejo de pantalla con optimización de po- 
sicionamiento de cursor. 
cc ¡flags fichero -1curses -ltermcap [librer 
-lcurses rutinas de la librería Cirses 
-ltermcap rutinas de la librería termcap 


CUSERID - devuelve el nombre del usuario conectado al terminal. 


include <stdio.h> 
char *Xcuserid(s) 
char *s; 


"ECVT, FCVT, GCVT - realiza la conversión de un número en coma 
flotante a una cadena de caracteres. 
char *kecvt (value, ndigit, decpt, sign) 


double value; 
int mdigit, *decpt, *Xsign; 


o 
char Xfcvt(value, ndigit, decpt, sign) 
double value; 
int ndigit, Xdecpt, *Xsign; 

lo) 


char Xgcvt (value, ndigit, buf) 
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double value: 
int ndigit; 
char *Xbuf;z 


FABS - devuelve el valor absoluto de un número. 


include <math.h> 
double fabs(x) 
double x; 


FCLOSE, FFLUSH - cierra un canal de E/S o vacía su buffer. 


include <stdio.h> 
int fcloseílstream) 
FILE *xstream; 


*tinclude <stdio.h> 
int fflushístream); 
FILE *stream; 


FDOPEN - asocia un canal de E/S con un descriptor de fichero. 


Htinclude <stdio.h> 

FILE Xfdopen (fildes, type) 
int fildes; 

char *Xtype; 


FEOF - devuelve un valor distinto de cero cuando alcanza el final 
de fichero. 

include <stdio.h> 

int feof (stream) 

FILE Xstream; 


FERROR - devuelve un valor distinto de cero tras un error de lec- 
tura/escritura. 


tinclude <stdio.h> 
int ferror (stream); 
FILE *Xstream; 


FGETS - lee (n-1) caracteres de un canal de E/S. 


tinclude <¿stdio.h> 

char *fgetsís, n, stream) 

char *Xs; 

int n; a . 
FILE Xstream; 


PILENO - devuelve el número entero asociado al descriptor de fi- 
chero. 


include <stdio.h> 
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int filenoístream); 
FILE Xstream; 


FLOOR - ante un valor numérico con decimales devuelve el en- 
tero inmediatamente superior al representado sin decimales. 
include <math.h> 
double floor (x) 
double x;5 


FMOD - devuelve la función resto módulo. 


*+include <stdio.h> 
double fmod(x, y) 
double x, y5 


FOPEN - abre un fichero y lo asocia con un canal de E/S. 
include <stdio.h> 
FILE Xfopen (filename, type) 
char filename, *Xtype; 


FPRINTF - escribe salida con formato sobre un canal de E/5S. 
"  Hinclude <stdio.h> 
int fprintf (stream, format [,¿argl ...) 
FILE Xstream; 
char format; 


FPUTC - escribe un carácter en un canal de E/S. 
include <stdio.h> 
int fputcic, stream); 
FILE X*Xstream; 


FREAD, FWRITE - proporciona E/S binaria con buffer. t 
include <stdio.h> 
int fread(ptr, size, nitems, stream) 
chat *Xptr; 
int size, nitems; 
FILE *stream; 


tinclude <stdio.h> 

int fwritelptr, size, nitems, stream) 
char *ptr; 

int size, nitems; 

FILE *Xstream; 


FREOPEN - sustituye un canal de E/S abierto por un fichero. . 


include <stdio.h> 
FILE *Xfreopen(filename, type, stream); 
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char *filename; 
FILE Xstream; 


FSCANF - lee datos de un canal de E/S. 


include <stdio.h> 

int fscanf(stream, format [, pointer] ...) 
FILE kstream; 

char *Xformat; 


FSEEK - posiciona el puntero para la próxima lectura/escritura en 
un canal de E/S. 

*tinclude <stdio.h> 

int fseek (stream, offset, ptrname) 

FILE Xstream; 

long offset; 

int ptrname; 


FTELL - devuelve la posición actual del puntero en un canal 
de E/S. 

*include <stdio.h> 

long ftell (stream); 

FILE *Xstream; 


GETC, GETCHAR, FGETC, GETW - toma un carácter o una pala- 
bra de un canal de E/S. 
Htinclude <stdio.h> 


int getcístream) 
FILE *Xstream; 


include <stdio.h> 
int getchar(); 


include <stdio.h> 
int getw(stream) 
FILE *kstream; 


GETENV - busca un nombre en el entorno de ejecución del 
usuario. : 


char kgetenv (name) 
char X*Xname; 


GETGRENT, GETGRGID, GETGRNAM, 'SETGRENT, ENDGRENT 
actúa sobre las protecciones de grupo de usuarics de un fichero. 
*tinclude <grp.h> 
struct group *Xgetgrent () 
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tinclude <grp-.h> 
struct group *Xgaetgrgid(gid) 


int gid; 
0) 
tinclude <grp.h> 
struct group *Xgetgrnam(name) 
char Xname; 
o) 
tinclude <grp.h> 
void setgrent() 
(e) 


*include <grp.h> 
void endgrent() 


GETLOGIN - devuelve el nombre del usuario conectado al ter- 
minal. 
char *Xgetlogin(); 


GETOPT - extrae las letras de opciones de un vector de argu- 
mentos. 

int getoptíargc, argv, optstring) 

int argc; 

char Akargv; 

char X*Xopstring; 

extern char *Xoptaros 

extern int optind; 


GETPASS - lee la clave de acceso (password) de un usuario. 
int getpw(uid, buf) 
int uid; 
char Xbuf; 


$ 


GETPWENT, GETPWUID, GETPWNAM, SETPWENT, ENDPWENT 
actúa sobre las entradas del fichero de usuarios (passwa). 


include <pwd.h> 
struct passwd' *Xgetpwent () 


0) 
tinclude <pwd.h> 
struct passwd *Xgetpwuid(uid) 
int uid; 

o) 


Hinclude <pwd.h> 
struct passwd kgetpwnam(name) 
char Xname; 
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tinclude <pwd.h> 
void setpwent () 


Hinclude <¿pwd.h> 
void endpwent () 


GETS - lee un literal de un canal de E/S. 
Hinclude ístdio.h> 
char *Xgets(s); 5. 
char *Xs; 


LOGNAME - devuelve el nombre de conexión (login) de un 
usuario. 
char *Xlogname() 


LONGJMP - recupera el stack del entorno de usuario salvado pre- 
viamente. 


tinclude <setjimp.h> 
void longjmplenv, val) 
jimp_buf env; 

int val; 


MALLOC, FREE, REALLOC, CALLOC - OS: dinámicos de 
memoria central, 


char *Xmalloc (size) 
unsigned size; 


0) 
void freeífptr) 
char *Xptr; 
0) 
char *realloc (ptr, size) 
char *ptr; 
unsigned size; 
o) 


char *calloc(nelem, elsize) 
unsigned nelem, elsize; 


METEMP - crea un nombre de fichero garartizando que sea único. 
char *Xmktemp (template) 
char *Xtemplate; » 


PERROR, ERRNO, SYS_ERRLIST, SYS_NERROR - - trata los mensajes 
de error del sistema. 


void perror (s) 
char Ys; 
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. 


extern int errno; 


extern char Xsys_errlist(Cl; 
o 
extern int sys_nerr; 


POPEN, PCLOSE - inicia o finaliza la E/S de un proceso a través 
de un pipe (tubería). 
Htinclude <¿stdio.h> 


FILE *popen (command, type) 
char *command, Xtype; 


tinclude <¿stdio.h> 
int pclose(stream) 
FILE *Xstream; 


PRINTF - imprime salida con formato. 
tinclude <stdio.h> 
int printf(format [,argl +...) 
char *Xformat; 


PUTC - pone un carácter en un canal de E/S. 


tinclude <stdio.h> 
int putcíc, stream) 
char 3 

FILE *Xstream; 


PUTCHAR - pone un carácter en el fichero de salida estándar 
(stdout), por defecto la pantalla. 


tinclude <stdio.h> 
int putchar (c) 
char c;5 


t 


PUTS, FPUTS - pone un literal en un canal de E/S. 
tinclude <stdio.h> 
int puts(s) 
char Xs; 
0) £ 
include <¿stdio.h> 
int fputsís, stream) 
char *Xs; 
FILE Xstream; 


PUTW - pone una palabra en un canal de E/S. 
tinclude ¿stdio.h> 
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int putwíw, stream) 
int w; 
FILE kXstream; 


RAND, SRAND - genera números aleatorios. 
int rand() 


o) 


void srand(seed) 
unsigned seed; sm. 


REWIND - posiciona al principio el puntero de acceso a un fichero. 
Htinclude ¿stdio.h> 
void rewind(stream) 
FILE X*Xstream; 


SCANF - lee del canal de entrada estándar (stdin). 


Hinclude <stdio.h> 
int scanf(íformat [,pointer] ...) 
char *Xformat; 


SETBUF - asigna un buffer a un canal de E/S. 
*tinclude <stdio.h> 
void setbuf (stream, buf) 
FILE *Xstream; 
char *Xbuf; 


SET]MP - guarda el stack de ejecución de un proceso. 
include <setimp.h> 
int setjmp (env) 
jump_buf env; 


SLEEP - suspende la ejecución de un proceso durante un interva- 
lo de tiempo especificado. 


unsigned sleep (seconds) 
unsiganed seconds; 


SPRINTF - escribe en un literal con salida formateada. 


include <stdio.h> 
int sprintf(s, format [,argl ...) 
char *Xs, *Xformat; . 


SSCANF - lee un literal con formato. 


Hinclude <stdio.h> 
int sscanf(s, format [,pointer] ...) 
char *Xs, *Xformat; 
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o O 


STDIO - E/S estándar bufferizada. 


include <stdio.h> 
FILE *Xstdin, *Xstdout, Xstderr; 


STRING - realiza operaciones con literales (strings). 


STRCAT - añade una copia del literal s2 al final del literal sl 
include <stdio.h> 
char Xstrcat (si, s2) 
char Xsi, *Xs2; 


STRNCAT - añade una copia de “n” caracteres del literal s2 al final 
del literal sl 

tinclude <stdio.h> 

char *Xstrncat (sl, s2, n) 

char *si, *s2; 

int n; 


STRCMP - compara los literales sl y sa 
include <stdio.h> 
int strecmpísi, s2) 
char *Xs1, %*s2; 


STRNCMP - compara al menos “n” caracteres de los literales s2 y 
sl 

include ¿string.h> 

int strncmp(si, s2, n) 

char *Xsi, %s2; 

int n; 


STRCPY - copia el literal s2 sobre el literal sl 


Htinclude <string.h> 
char X*Xstrcpy(sl, s2) 
char X*Xs1, %Xs2;3 


STRNCPY - copia exactamente “n” caracteres del literal s2 al lite- 
ral sl 

tinclude <string.h> 

char Aistrncpy(ísi, s2, n) 

char Xsl, *Xs2; 

int n; 


STRLEN - devuelve el número de caracteres distintos de nulo 


+tinclude ¿string.h> 
int strlen(s) 
char *s; 
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STRCHR - devuelve un puntero a la primera ocurrencia del carác- 
ter "c” en el literal "s” 

*tinclude <string.h> 

char Xstrchr(s, Cc) 


char *Xs, €; 


STRRCHR - devuelve un puntero a la última ocurrencia del carác- 
ter *c” en el literal “s” 

include <string.h> 

char *Xstrrchr(s, Cc) *. 


char *Xs, Cc; 


STRPBRK - devuelve un puntero al primer carácter del literal s] 
encontrado en el literal s2 


*+include <stdio.h> 
char Xstrpbrk (si, s2) 
char *si, Xs2; 


STRSPN - devuelve la longitud del literal s1 que coincide con el 
literal s2 

*include <string.h> 

int strspnísi, s2) 

char Xsi, *Xs2; 


STRCSPN - devuelve la longitud del literal s1 que no coincide con 
el literal s2 

*include <string.h> 

int strcspnísi, s2) 

char *si, *s2; 


STRTOK - devuelve un puntero a la primera ocurrencia del literal 
sl en el literal s2 

tinclude <string.h> 

char X*Xstrtok (sli, s2) 

char *s1, Xs2; 


STRTOL, ATOL, ATOI - convierten un literal en un entero 


long strtol (str, ptr, base) 

char *Xstr; 

char *kptr;3; 

int base; » 


long atol (str) 
char Xstr; 


int atoi (str) 
char *Xstr;3 
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SWAB - intercambia bytes. 
void swabífrom, to, nbytes) 
char *Xfrom, *to; 
int nbytes; 


SYSTEM - proporciona un comando de la shell. 


include <¿stdio.h> 
int system(string) 
char *Xstring;3 


TERMCAP - proporciona subrutinas para manejo del terminal, es- 
pecificando las características de los atributos de vídeo propios 
de cada terminal. . 


TEGENT - pone el nombre del terminal en un buffer 


tgetent (bp, name) 
char *Xbp, Xname; 


TGETNUM - devuelve el valor numérico del atributo “id” del ter- 
minal : 


Htgetnumtid) 
char Xid; 


TGETSTR - toma el valor literal del atributo “id” del terminal 


char *k tgetstr(id, area) 
char *Xid, *X*Xarea; 3 


TGOTO - devuelve un literal para efectuar el posicionamiento del 
cursor del terminal 


char *Xtgotoícm, destcol, destline) 
char *Xcm; 
int destcol, destline; 


TMPFILE - crea un fichero temporal. 


include <stdio.h> 
FILE *Xtmpfile(); 


TMPNAME, TEMPNAM - asigna un nombre a un fichero temporal. 


*include <stdio.h> 
char *Ximpnamí(s) 
char X*Xs; 


tinclude <stdio.h> 
char *Xtempnam(dir, pfx) 
char *Xdir, *ptxs3 
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TTYNAME, ISATTY - proporciona el nombre del terminal. 


char Xttyname(fildes) 
int fildes; 


int isattyífiles) 
int fildes; 


UNGETC - devuelve un carácter a un canal de E/S. 


tinclude <stdio.h> o 
int ungetc(c, stream) 

char €; 

FILE *stream; 
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BIBLIOTECA BASICA INFORMATICA 


INDICE GENERAL 


Dentro y fuera del ordenador 
Todo lo que debemos saber para poder comprender 
en qué consisten y cómo funcionan los ordenadores. 


2 Diccionario de términos informáticos 


Una perfecta guía en ese «maremagnum» de palabras y 
frases ininteligibles que se usan en Informática. 


3 Cómo elegir un ordenador... que se ajuste a nuestras 


7 


8 


9 


necesidades 

Las características y detalles en los que deberemos 
centrar nuestra atención a la hora de elegir un 
ordenador. 

Cuidados del ordenador... cosas que debemos hacer o 
evitar 

Esos consejos que le evitarán problemas con su 
equipo, permitiéndole obtener el máximo provecho. 
¡Y llegó el BASIC! (1) 

Un claro y sencillo acercamiento a los principios de 
este popular lenguaje. 

Dimensión MSX 

El primer BASIC estándar que ha conseguido difundirse 
de verdad no es sólo un lenguaje; hay bastante más. 
¡Y llegó el BASIC! (I) 

Instrucciones y comandos que quedaron por explicar 
en el la parte 1. 

Introducción al Pascal 

Una buena manera de adentrarse en la programación 
estructurada, ¡la nueva ola de la Informática! 
Programando como es debido... algoritmos y otros 
elementos necesarios. 

Ideas para mejorar la funcionalidad y desarrollo de sus 
programas. 
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13 


14 


15 


16 


17 


18 


19 


20 


21 


22 


23 


24 


Sistemas operativos y software de base 

Qué son, para qué sirven. Unos desconocidos muy 
importantes. 

Sistema operativo CP/M 

Uno de los sistemas operativos para microprocesadores 
de 8 bits de mayor difusión en el mercado. 

MS-DOS: el estándar de IBM 

Sistema operativo para el microprocesador de 16 bits 
8088, adoptado por el IBM-PC, 

Paquetes de aplicaciones. Software “pret a porter” 
Características y peculiaridades de los más importantes 
paquetes de aplicaciones. 

VisiCalc: una buena hoja de cálculo 

Interioridades y manejo de una de las hojas de cálculo 
más usadas. 

Dibujar con el ordenador 

Profundizando en una de las facetas útiles y divertidas 
que nos ofrecen los ordenadores. 

Tratamiento de textos... para escribir con el ordenador 
Cómo convertir su ordenador en una máquina de 
escribir con memoria y todo tipo de posibilidades. 
Diseño de juegos 

Particularidades características de esta aplicación de 
los ordenadores. 

LOGO: la tortuga inteligente 

Un lenguaje conocido por su «cursor gráfico», la.tortuga, 
y sus aplicaciones pedagógicas al alcance de su mano. 
Paquetes integrados: Lotus 1-2-3 y Simphony 

Estudio de dos de los paquetes integrados (Hoja de 
cálculo+base de datos+..) más conocidos. 

dBASE II y dBASE II 

Cómo aprovechar las dos versiones más recientes de 
esta importante base de datos. 

Bancos de datos (1) 


Peculiaridades de una de las aplicaciones de los 


ordenadores más interesantes y que más dinero 
mueven. 

Bancos de datos (II) 

Profundizando en sus características. 

FORTH: anatomía de un lenguaje inteligente 3 
Principales características de un lenguaje moderno, 
flexible y de amplio uso, en la robótica. 

BASIC y tratamiento de imágenes 

Todo lo que en ¡Y llegó el BASIC! no se pudo ver sobre 
las imágenes y gráficos en el BASIC, 


25 Los ordenadores uno a uno 
Un amplio y completo estudio comparativo. 
26 Cálculo numérico en BASIC 
Una aplicación especializada a su disposición. 
27 Multiplan 
Cómo hacer uso de este moderno paquete de 
aplicaciones. : 
28 FORTRAN y COBOL 
Dos lenguajes muy especializados y distintos. 
29 Softest. Los programas a examen 
30 Cómo realizar nuestro propio banco de datos 
Conocimientos necesarios para poder fabricar un 
banco de datos a nuestro gusto y medida. 


NOTA: Ingelek, S. A. se reserva el derecho de modificar, sin 
previo aviso, el orden, título o contenido de cualquier 
volumen de la colección. 


uando oímos hablar de lenguajes de 
programación los que se nos vienen a la 
memoria son el BASIC, el FORTRAN, el 
PASCAL u otros. Sin embargo, el C se ha 
ganado en poco tiempo el derecho a en- 
trar en esta lista de “habituales”. Son po- 
cas las personas que se mueven alrede- 
dor de los ordenadores personales y su- 
permicros que no han oído hablar de este lenguaje y de 
sus posibilidades. 

Una de las particularidades que más le ha elevado a la 
“cresta de la ola” es su íntima ligazón con el sistema ope- 
rativo Unix, que se está convirtiendo en un estándar para 
los supermicros de 16, 32 y, en el futuro, 64 bits. 

Este volumen de la B.B.I. pretende ser una primera intro- 
ducción al C. dando una primera idea de su desarrollo, 
características y evolución futura, de forma que queden 
sentadas las bases que faciliten el acceso a libros más es- 
pecializados 
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